亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C#中Invoke的用法講解

 更新時間:2021年07月08日 09:41:29   作者:三小  
這篇文章主要介紹了C#中Invoke的用法講解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下

C#中Invoke的用法()

invoke和begininvoke 區(qū)別

一直對invoke和begininvoke的使用和概念比較混亂,這兩天看了些資料,對這兩個的用法和原理有了些新的認識和理解。

首先說下,invoke和begininvoke的使用有兩種情況:

1. control中的invoke、begininvoke。

2. delegrate中的invoke、begininvoke。

這兩種情況是不同的,我們這里要講的是第1種。下面我們在來說下.NET中對invoke和begininvoke的官方定義。

control.invoke(參數(shù)delegate)方法:在擁有此控件的基礎窗口句柄的線程上執(zhí)行指定的委托。

control.begininvoke(參數(shù)delegate)方法:在創(chuàng)建控件的基礎句柄所在線程上異步執(zhí)行指定委托。

根據(jù)這兩個概念我們大致理解invoke表是同步、begininvoke表示異步。

如果你的后臺線程在更新一個UI控件的狀態(tài)后不需要等待,而是要繼續(xù)往下處理,那么你就應該使用BeginInvoke來進行異步處理。

如果你的后臺線程需要操作UI控件,并且需要等到該操作執(zhí)行完畢才能繼續(xù)執(zhí)行,那么你就應該使用Invoke。

我們來做一個測試。

invoke 例子:

private void button1_Click(object sender, EventArgs e)
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"AAA");
            invokeThread = new Thread(new ThreadStart(StartMethod));
            invokeThread.Start();
            string a = string.Empty;
            for (int i = 0; i < 3; i++)      //調整循環(huán)次數(shù),看的會更清楚
            {
                Thread.Sleep(1000);   
                a = a + "B";
            }
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+a);
}

 private void StartMethod()
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"CCC");
            button1.Invoke(new invokeDelegate(invokeMethod));  
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"DDD");
}

 private void invokeMethod()
{
            //Thread.Sleep(3000);
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString() + "EEE");
}

結論:我們運行后,看下程序的運行順序,1AAA->3CCC和1BBB->1EEE ->3DDD 。

解釋:主線程運行1AAA,然后1BBB和子線程3CCC同時執(zhí)行,然后通過invoke來將invokemethod方法提交給主線程,然后子線 程等待主線程執(zhí)行,直到主線程將invokemethod方法執(zhí)行完成(期間必須等待主線程的任務執(zhí)行完成,才會去執(zhí)行invoke提交的任務),最后執(zhí) 行子線程3DDD。

begininvoke 例子:

private void button1_Click(object sender, EventArgs e)
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"AAA");
            invokeThread = new Thread(new ThreadStart(StartMethod));
            invokeThread.Start();
            string a = string.Empty;
            for (int i = 0; i < 3; i++)      //調整循環(huán)次數(shù),看的會更清楚
            {
                Thread.Sleep(1000);   
                a = a + "B";
            }
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+a);
}

 private void StartMethod()
{
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"CCC");
            button1.BeginInvoke(new invokeDelegate(invokeMethod));  
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString()+"DDD");
}

 private void beginInvokeMethod()
        {
            //Thread.Sleep(3000);
            MessageBox.Show(Thread.CurrentThread.GetHashCode().ToString() + "EEEEEEEEEEEE");
        }

結論: 我們運行后看看執(zhí)行的結果:1AAA->1BBB和3CCC->1EEE和3DDD。

解釋: 主線程運行1AAA,然后1BBB和子線程3CCC同時執(zhí)行,然后通過begininvoke來將invokemethod方法提交給主線程,然后主線程執(zhí)行1EEE(主線程自己的任務執(zhí)行完成), 同時子線程繼續(xù)執(zhí)行3DDD。

通過這個兩段代碼的測試比較,我們會發(fā)現(xiàn)其實invoke和begininvoke所提交的委托方法都是在主線程中執(zhí)行的,其實根據(jù)我invoke 和begininvoke的定義我們要在子線程中來看這個問題,在invoke例子中我們會發(fā)現(xiàn)invoke所提交的委托方法執(zhí)行完成后,才能繼續(xù)執(zhí)行 DDD;在begininvoke例子中我們會發(fā)現(xiàn)begininvoke所提交的委托方法后,子線程講繼續(xù)執(zhí)行DDD,不需要等待委托方法的完成。 那么現(xiàn)在我們在回想下invoke(同步)和begininvoke(異步)的概念,其實它們所說的意思是相對于子線程而言的,其實對于控件的調用總是由 主線程來執(zhí)行的。我們很多人搞不清這個同步和異步,主要還是因為我們把參照物選錯了。其實有時候光看概念是很容易理解錯誤的。

解決從不是創(chuàng)建控件的線程訪問它

在多線程編程中,我們經(jīng)常要在工作線程中去更新界面顯示,而在多線程中直接調用界面控件的方法是錯誤的做法,Invoke 和 BeginInvoke 就是為了解決這個問題而出現(xiàn)的,使你在多線程中安全的更新界面顯示。

正確的做法是將工作線程中涉及更新界面的代碼封裝為一個方法,通過 Invoke 或者 BeginInvoke 去調用,兩者的區(qū)別就是一個導致工作線程等待,而另外一個則不會。

而所謂的“一面響應操作,一面添加節(jié)點”永遠只能是相對的,使 UI 線程的負擔不至于太大而已,因為界面的正確更新始終要通過 UI 線程去做,我們要做的事情是在工作線程中包攬大部分的運算,而將對純粹的界面更新放到 UI 線程中去做,這樣也就達到了減輕 UI 線程負擔的目的了。

舉個簡單例子說明下使用方法,比如你在啟動一個線程,在線程的方法中想更新窗體中的一個TextBox..

using System.Threading; 

//啟動一個線程 
Thread thread=new Thread(new ThreadStart(DoWork)); 
thread.Start(); 

//線程方法 
private void DoWork() 
{ 
this.TextBox1.Text="我是一個文本框"; 
} 

如果你像上面操作,在VS2005或2008里是會有異常的... 

正確的做法是用Invoke\BeginInvoke

using System.Threading;
namespace test
{
public partial class Form1 : Form
{
public delegate void MyInvoke(string str1,string str2);
public Form1()
{
InitializeComponent();


}
public void DoWork()
{
MyInvoke mi = new MyInvoke(UpdateForm);
this.BeginInvoke(mi, new Object[] {"我是文本框","haha"});
}
public void UpdateForm(string param1,string parm2)
{
this.textBox1.Text = param1+parm2;
}
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(DoWork));
thread.Start();
}
}
}

注意代理的使用!

后面再次補充 在 WinForm開發(fā)過程中經(jīng)常會用到線程,有時候還往往需要在線程中訪問線程外的控件,比如:設置textbox的Text屬性等等。如果直接設置程序必 定會報出:從不是創(chuàng)建控件的線程訪問它,這個異常。通常我們可以采用兩種方法來解決。一是通過設置control的屬性。二是通過delegate,而通 過delegate也有兩種方式,一種是常用的方式,另一種就是匿名方式。下面分別加以說明.

首先,通過設置control的一個屬性值為false.我們可以在Form_Load方法中添加:Control.CheckForIllegalCrossThreadCalls=false;來解決。設置為false表示不對錯誤線程的調用進行捕獲。這樣在線程中對textbox的Text屬性進行設置時就不會再報錯了。

其次,通過delegate的方法來解決。

普通的委托方法例如:

delegate void SafeSetText(string strMsg);
private void SetText(string strMsg)
{
 if(textbox1.InvokeRequired)
 {
            SafeSetText objSet=new SafeSetText(SetText);
            textbox1.Invoke(objSet,new object[]{strMsg});

 }
 else
 {
   textbox1.Text=strMsg;
 }
}

在線程內需要設置textbox的值時調用SetText方法既可。我們還可以采用另一種委托的方式來實現(xiàn),那就是匿名代理,例如:

delegate void SafeSetText(string strMsg);
private void SetText2(string strMsg)
{
  SafeSetText objSet = delegate(string str)
   {
       textBox1.Text = str;
   }
   textBox1.Invoke(objSet,new object[]{strMsg});
}

這樣同樣可以實現(xiàn)。

個人覺得還是采用代理好些。

在C# 3.0及以后的版本中有了Lamda表達式,像上面這種匿名委托有了更簡潔的寫法。.NET Framework 3.5及以后版本更能用Action封裝方法。例如以下寫法可以看上去非常簡潔:

void ButtonOnClick(object sender,EventArgs e)
{
    this.Invoke(new Action(()=>
    {
        button.Text="關閉";
    }));
}

最新:

Invoke(() =>
{
 button.Text="關閉";
});

到此這篇關于C#中Invoke的用法講解的文章就介紹到這了,更多相關C# Invoke內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • C#基于Miniblink控件編寫一個簡易的瀏覽器

    C#基于Miniblink控件編寫一個簡易的瀏覽器

    miniblink是一款精簡小巧的瀏覽器控件,基于chromium精簡而成,是市面上最小巧的chromium內核控件沒有之一,本文將結合C#和Miniblink編寫一個簡易的瀏覽器,感興趣的可以了解下
    2024-01-01
  • c#調用winrar解壓縮文件代碼分享

    c#調用winrar解壓縮文件代碼分享

    這篇文章主要介紹了c#調用winrar解壓縮文件的方法,大家參考使用吧
    2013-12-12
  • UGUI實現(xiàn)圖片拖拽功能

    UGUI實現(xiàn)圖片拖拽功能

    這篇文章主要為大家詳細介紹了UGUI實現(xiàn)圖片拖拽功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-02-02
  • 動態(tài)改變gridview列寬度函數(shù)分享

    動態(tài)改變gridview列寬度函數(shù)分享

    通常用GridView綁定datatable,由于需要動態(tài)綁定到不同的datatable所以需要動態(tài)調整GridView的寬度。寫了這個函數(shù)實現(xiàn)該功能
    2014-01-01
  • C#檢查foreach判讀是否為null的方法

    C#檢查foreach判讀是否為null的方法

    這篇文章主要介紹了C#如何檢查foreach判讀其是否為null,文中給出了示例代碼,介紹的很詳細,需要的朋友可以參考下方法
    2016-09-09
  • Unity常用音頻操作類示例代碼

    Unity常用音頻操作類示例代碼

    這篇文章主要介紹了Unity常用音頻操作類,本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-07-07
  • C#網(wǎng)絡編程之Socket編程

    C#網(wǎng)絡編程之Socket編程

    本文詳細講解了C#網(wǎng)絡編程的Socket編程,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-02-02
  • C# WinForm應用程序降低系統(tǒng)內存占用方法總結

    C# WinForm應用程序降低系統(tǒng)內存占用方法總結

    這篇文章主要介紹了C# WinForm應用程序降低系統(tǒng)內存占用方法總結,本文總結了9個方法,同時給出了一個定期清理執(zhí)行垃圾回收代碼,需要的朋友可以參考下
    2014-10-10
  • C#實現(xiàn)圖片上傳(PC端和APP)保存及 跨域上傳說明

    C#實現(xiàn)圖片上傳(PC端和APP)保存及 跨域上傳說明

    這篇文章主要介紹了C#實現(xiàn)圖片上傳(PC端和APP)保存及 跨域上傳說明的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-12-12
  • c# 模擬線性回歸的示例

    c# 模擬線性回歸的示例

    這篇文章主要介紹了c# 模擬線性回歸的示例,幫助大家利用c#進行機器學習,感興趣的朋友可以了解下
    2020-10-10

最新評論