Có gì mới?

Chào mừng bạn đến với diễn đàn Tui Học Web

Vui lòng đăng ký hoặc đăng nhập để sử dụng hết các tính năng của forum nhé!
  • Thông báo: Hiện tại do admin không có nhiều thời gian và không thể tiếp tục quản trị và maintain forum, do đó forum sẽ có dự định được đóng lại từ 06/2022, cảm ơn mọi người đã đồng hành cùng forum suốt thời gian vừa qua. Cảm ơn rất nhiều.

Kiến thức Invoke là gì? cách sử dụng phương thức Invoke()

Kiến thức

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]

STAThread chính là entrypoint

Đó là thread chính (main-thread).
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!​


 
shape1
shape2
shape3
shape4
shape7
shape8
Bên trên