Invoke là gì? cách sử dụng phương thức Invoke()
Xin chào các bạn, bài viết hôm nay mình sẽ tiếp tục hướng dẫn các bạn về Invoke trong lập trình C#, Winform.Khi các bạn lập trình bất động bộ, sử dụng Task, thì các bạn thường hay gặp về phương thức Invoke.
Vậy Invoke C# là gì?
Khi ứng dụng của bạn chạy, có một thread được tạo ra để chạy hàm Main().
Ở trên hàm Main() có [STAThread]
Nếu chương trình của bạn có nhiều thread thực hiện các tác vụ xử lý khác và các thread này cần sử dụng tài nguyên từ thread chính thì bạn phải cần tới Invoke.
Thực ra, bạn có thể đặt thuộc tính CheckForIllegalCrossThreadCalls = false; cho form (hoặc control) và sử dụng các tài nguyên từ thread khác một cách thoải mái.
Nhưng như vậy, chương trình của bạn sẽ rơi vào trạng thái ko an toàn (unsafe) và sẽ bị crash bất cứ lúc nào khi mà các thread tranh chấp tài nguyên với nhau.
C# cung cấp 1 giải pháp an toàn hơn đó là Invoke. Khi bạn gọi phương thức này của một form (hoặc control) từ một thread khác, form (control) đó sẽ bị lock, chỉ cho phép thread đã gọi nó truy cập. Khi thread này hoàn thành tác vụ của nó, form (control) lại được giải phóng cho thread khác gọi.
Như vậy, các thread sẽ được đồng bộ với nhau và chương trình của bạn sẽ ko bị crash. Đó gọi là thread-safe.
Có những control ko yêu cầu Invoke để thực hiện thread-safe. Nghĩa là nó có thể được truy cập một cách trực tiếp ko qua Invoke.
Thuộc tính InvokeRequired sẽ cho biết một control có yêu cầu Invoke khi gọi hay ko ?
- Khi gọi Invoke, bạn phải truyền cho nó một delegate. Bạn có thể sử dụng delegate MethodInvoker có sẵn của C#.
Ví dụ: Chương trình của mình có một listbox, mình sẽ tạo một thread mới.
Thread này có nhiệm vụ thêm các số từ 1-9999 vào listbox, đồng thời cập nhật tiến độ qua một progressbar.
C#:
private void btnCreateNumber_Click(object sender, EventArgs e)
{
lstNumber.Items.Clear();
// Tạo và chạy thread
Thread thrGenerating = new Thread(new ThreadStart(DoWork));
thrGenerating.Start();
}
private void DoWork()
{
for (int i = 1; i <= 9999; i++)
{
// Thêm item vào list qua invoke
lstNumber.Invoke(new Action(()=>
{
lstNumber.Items.Add(i);
lstNumber.TopIndex = lstNumber.Items.Count - 1;
}));
// Cập nhật tiến độ qua progress bar
pgrOperation.Invoke(new Action(() =>
{
pgrOperation.Value = (i * 100 / 9999);
}));
}
MessageBox.Show("Finish!");
}
Chúc các bạn thành công!