<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        .net面向對象之多線程(Multithreading)及 多線程高級應用

        來源:懂視網 責編:小采 時間:2020-11-27 22:38:24
        文檔

        .net面向對象之多線程(Multithreading)及 多線程高級應用

        .net面向對象之多線程(Multithreading)及 多線程高級應用:在.net面向對象程序設計階段在線程資源共享中的線程安全和線程沖突的解決方案;多線程同步,使用線程鎖和線程通知實現線程同步,具體內容介紹如下: 1、 ThreadStatic特性 特性:[ThreadStatic] 功能:指定靜態字段在不同線程中擁有不同的值 在此之前,我們
        推薦度:
        導讀.net面向對象之多線程(Multithreading)及 多線程高級應用:在.net面向對象程序設計階段在線程資源共享中的線程安全和線程沖突的解決方案;多線程同步,使用線程鎖和線程通知實現線程同步,具體內容介紹如下: 1、 ThreadStatic特性 特性:[ThreadStatic] 功能:指定靜態字段在不同線程中擁有不同的值 在此之前,我們

        在.net面向對象程序設計階段在線程資源共享中的線程安全和線程沖突的解決方案;多線程同步,使用線程鎖和線程通知實現線程同步,具體內容介紹如下:

        1、 ThreadStatic特性

        特性:[ThreadStatic]

        功能:指定靜態字段在不同線程中擁有不同的值

        在此之前,我們先看一個多線程的示例:

        我們定義一個靜態字段:

         static int num = 0;
         然后創建兩個線程進行分別累加:

        new Thread(() =>
        {
         for (int i = 0; i < 1000000; i++)
         ++num;
         Console.WriteLine("來自{0}:{1}", Thread.CurrentThread.Name, num);
        })
        { Name = "線程一" }.Start(); 
        
        new Thread(() =>
        {
         for (int i = 0; i < 2000000; i++)
         ++num;
         Console.WriteLine("來自{0}:{1}", Thread.CurrentThread.Name, num);
        })
        { Name = "線程二" }.Start();
        
        

        運行多次結果如下:

            

        可以看到,三次的運行結果均不相同,產生這種問題的原因是多線程中同步共享問題導致的,即是多個線程同時共享了一個資源。如何解決上述問題,最簡單的方法就是使用靜態字段的ThreadStatic特性。

        在定義靜態字段時,加上[ThreadStatic]特性,如下:

        代碼如下:
         [ThreadStatic]
        static int num = 0;
        兩個線程不變的情況下,再次運行,結果如下:

         

        不論運行多少次,結果都是一樣的,當字段被ThreadStatic特性修飾后,它的值在每個線程中都是不同的,即每個線程對static字段都會重新分配內存空間,就當然于一次new操作,這樣一來,由于static字段所產生的問題也就沒有了。

        2. 資源共享

        多線程的資源共享,也就是多線程同步(即資源同步),需要注意的是線程同步指的是線程所訪問的資源同步,并非是線程本身的同步。

        在實際使用多線程的過程中,并非都是各個線程訪問不同的資源。

        下面看一個線程示例,假如我們并不知道線程要多久完成,我們等待一個固定的時間(假如是500毫秒):

        先定義一個靜態字段:

         static int result;
        創建線程:

        Thread myThread = new Thread(() =>
        {
         Thread.Sleep(1000);
         result = 100;
        });
        myThread.Start();
        Thread.Sleep(500); 
        Console.WriteLine(result);
        

        運行結果如下:

         

        可以看到結果是0,顯然不是我們想要的,但往往在線程執行過程中,我們并不知道它要多久完成,能不能在線程完成后有一個通知?

        這里有很多笨的方法,比如我們可能會想到使用一個循環來檢測線程狀態,這些都不是理想的。

        .NET為我們提供了一個Join方法,就是線程阻塞,可以解決上述問題,我們使用Stopwatch來記時,

        改進線程代碼如下:

        System.Diagnostics.Stopwatch watch = System.Diagnostics.Stopwatch.StartNew();
        Thread myThread = new Thread(() =>
        {
         Thread.Sleep(1000);
         result = 100;
        });
        myThread.Start();
        Thread.Sleep(500);
        myThread.Join();
        Console.WriteLine(watch.ElapsedMilliseconds);
        Console.WriteLine(result);
        
        

        運行結果如下:

         結果和我們想要的是一致的。

        3. 線程鎖

        除了上面示例的方法,對于線程同步,.NET還為我們提供了一個鎖機制來解決同步,再次改進上面示例如下:

        先定義一個靜態字段來存儲鎖:

        static object locker = new object();
        這里我們可以先不用考慮這個對象是什么。繼續看改進后的線程:

        System.Diagnostics.Stopwatch watch = System.Diagnostics.Stopwatch.StartNew();
        Thread t1 = new Thread(() =>
        {
         lock (locker)
         {
         Thread.Sleep(1000);
         result = 100;
         }
        });
        t1.Start();
        Thread.Sleep(100);
        lock (locker)
        {
         Console.WriteLine("線程耗時:"+watch.ElapsedMilliseconds);
         Console.WriteLine("線程
        輸出:"+result); }

        運行結果如下:

        運行結果和上面示例一樣,如果線程處理過程較復雜,可以看到耗時明顯減少,這是一種用比阻塞更效率的方式完成線程同步。

        4. 線程通知

        前面說到了能否在一個線程完成后,通知等待的線程呢,這里.NET為我們提供了一個事件通知的方法來解決這個問題。

        4.1 AutoResetEvent 

        先定義一個通知對象

        代碼如下:
         static EventWaitHandle tellMe = new AutoResetEvent(false);

        改進上面的線程如下:

        System.Diagnostics.Stopwatch watch = System.Diagnostics.Stopwatch.StartNew();
        Thread myThread = new Thread(() =>
        {
         Thread.Sleep(1000);
         result = 100;
         tellMe.Set();
        });
        myThread.Start();
        tellMe.WaitOne();
        Console.WriteLine("線程耗時:" + watch.ElapsedMilliseconds);
        Console.WriteLine("線程
        輸出:" + result);

         運行結果如下:

        4.2 ManualResetEvent

        和AutoResetEvent 相對的還有一個 ManualResetEvent 手動模式,他們的區別在于,在線程結束后ManualResetEvent 還是可以通行的,除非手動Reset關閉。下面看一個示例:

        先定義一個手動通知的對象:

        static EventWaitHandle mre = new ManualResetEvent(false);


        創建線程:

        System.Diagnostics.Stopwatch watch = System.Diagnostics.Stopwatch.StartNew();
        Thread myThreadFirst = new Thread(() =>
        {
         Thread.Sleep(1000);
         result = 100;
         mre.Set();
        }) { Name = "線程一" };
        Thread myThreadSecond = new Thread(() =>
        {
         mre.WaitOne();
         Console.WriteLine(Thread.CurrentThread.Name + "獲取結果:" + result + "("+System.DateTime.Now.ToString()+")");
        }) { Name="線程二"};
        myThreadFirst.Start();
        myThreadSecond.Start();
        mre.WaitOne();
        Console.WriteLine("線程耗時:" + watch.ElapsedMilliseconds + "(" + System.DateTime.Now.ToString() + ")");
        Console.WriteLine("線程
        輸出:" + result + "(" + System.DateTime.Now.ToString() + ")");

        運行結果如下:

        4.3. Semaphore

        Semaphore也是線程通知的一種,上面的通知模式,在線程開啟的數量很多的情況下,使用Reset()關閉時,如果不使用Sleep休眠一下,很有可能導致某些線程沒有恢復的情況下,某一線程提前關閉,對于這種很難預測的情況,.NET提供了更高級的通知方式Semaphore,可以保證在超多線程時不會出現上述問題。

        先定義一個通知對象的靜態字段:

        代碼如下:
           static Semaphore sem = new Semaphore(2, 2);

        使用循環創建100個線程:


        for (int i = 1; i <= 100; i++)
        {
         new Thread(() =>
         {
         sem.WaitOne();
         Thread.Sleep(30);
         Console.WriteLine(Thread.CurrentThread.Name+" "+DateTime.Now.ToString());
         sem.Release();
         }) { Name="線程"+i}.Start();
        }
        

        運行結果如下:

         

        可以看到完整的輸出我們所想要看到的結果。

        5. 本節要點:

        A.線程中靜態字段的ThreadStatic特性,使用該字段在不同線程中擁有不同的值

        B.線程同步的幾種方式,線程鎖和線程通知

        C.線程通知的兩種方式:AutoResetEvent /ManualResetEvent  和 Semaphore

        到此為止.net面向對象之多線程(Multithreading)及多線程高級應用介紹到此為止。

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        .net面向對象之多線程(Multithreading)及 多線程高級應用

        .net面向對象之多線程(Multithreading)及 多線程高級應用:在.net面向對象程序設計階段在線程資源共享中的線程安全和線程沖突的解決方案;多線程同步,使用線程鎖和線程通知實現線程同步,具體內容介紹如下: 1、 ThreadStatic特性 特性:[ThreadStatic] 功能:指定靜態字段在不同線程中擁有不同的值 在此之前,我們
        推薦度:
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top
        主站蜘蛛池模板: 亚洲成a人在线看天堂无码| 在线观看人成网站深夜免费| 亚洲伦乱亚洲h视频| 午夜在线免费视频| 免费人成在线观看视频播放| 极品色天使在线婷婷天堂亚洲| 国产在线a不卡免费视频| 黄色网址大全免费| 久久精品国产亚洲7777| 丁香花在线观看免费观看图片| 国产精品亚洲片在线观看不卡 | 免费视频爱爱太爽了| 亚洲午夜电影一区二区三区| 欧亚精品一区三区免费| 亚洲爆乳无码精品AAA片蜜桃| 永久免费bbbbbb视频| 国产AV日韩A∨亚洲AV电影| 亚洲午夜av影院| 久久午夜夜伦鲁鲁片免费无码| 亚洲欧洲国产精品你懂的| 最近高清中文字幕无吗免费看| 亚洲人成综合网站7777香蕉| 永久免费看bbb| aa在线免费观看| 亚洲成人福利在线观看| 国产精品无码素人福利免费| 国产区在线免费观看| 久久精品国产亚洲精品2020| 最近中文字幕mv手机免费高清| 妇女自拍偷自拍亚洲精品| 国产亚洲真人做受在线观看| 国产在线观看麻豆91精品免费 | 青青青国产手机频在线免费观看 | 国产亚洲福利一区二区免费看| 成人精品综合免费视频| 亚洲AV日韩AV永久无码免下载| 成人午夜视频免费| 永久在线观看免费视频| 亚洲а∨天堂久久精品9966| 亚洲人成亚洲人成在线观看| 国产一卡2卡3卡4卡无卡免费视频 国产一卡二卡3卡四卡免费 |