婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 簡單使用BackgroundWorker創建多個線程的教程

簡單使用BackgroundWorker創建多個線程的教程

熱門標簽:怎樣在地圖標注消火栓圖形 泰州手機外呼系統軟件 廈門四川外呼系統 百度地圖標注點擊事件 山東防封電銷卡辦理套餐 濟源人工智能電話機器人價格 杭州智能電話機器人 內蒙古智能電銷機器人哪家強 地圖標注位置多的錢

BackgroundWorker是一個非常不錯的線程控件,能避免界面假死,讓線程操作你想要做的事,它學習起來很簡單,但是能實現很強大的功能。發布這篇文章的目的是將最近學習到的共享出來,大家交流一下,當然我也是菜鳥,在這里你將學習到BackgroundWorker簡單使用,停止,暫停,繼續等操作,BackgroundWorker比起Thread和ThreadPool要簡單太多,為了更方便在實際應用中使用,我使用的是winform,沒有使用控制臺程序。

在UI界面里拖動一個button和richTextBox到界面。

我會從最簡單的開始,只有最簡單的代碼才會讓人有繼續學下去的欲望,下列代碼可以將1到999打印到richTextBox1控件上。

復制代碼 代碼如下:

private void button1_Click(object sender, EventArgs e)
 {
     //創建一個BackgroundWorker線程
     BackgroundWorker bw = new BackgroundWorker();
     //創建一個DoWork事件,指定bw_DoWork方法去做事
     bw.DoWork += new DoWorkEventHandler(bw_DoWork);
     //開始執行
     bw.RunWorkerAsync();
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     for (int i = 0; i 1000; i++)
     {
         this.richTextBox1.Text += i + Environment.NewLine;
     }
 }

但是很不幸,以上代碼會報錯,報錯信息:線程間操作無效: 從不是創建控件“richTextBox1”的線程訪問它。

那么我們繼續改造代碼,讓數字顯示在richTextBox1控件上,并且讓richTextBox1焦點處于最低端。

復制代碼 代碼如下:

private void button1_Click(object sender, EventArgs e)
 {
     //創建一個BackgroundWorker線程
     BackgroundWorker bw = new BackgroundWorker();
     //創建一個DoWork事件,指定bw_DoWork方法去做事
     bw.DoWork += new DoWorkEventHandler(bw_DoWork);
     //開始執行
     bw.RunWorkerAsync();
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     for (int i = 0; i 1000; i++)
     {
         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += i + Environment.NewLine;
         });
     }
 }

 private void richTextBox1_TextChanged(object sender, EventArgs e)
 {
     RichTextBox textbox = (RichTextBox)sender;

     textbox.SelectionStart = textbox.Text.Length;
     textbox.ScrollToCaret();
 }

上面是BackgroundWorker一個最簡單的例子,沒有多余復雜的代碼,這就是BackgroundWorker,下面我們加入停止按鈕,讓線程停下來。

再拖動一個button控件到界面,讓線程停止我們先要改造一下代碼,讓button事件也能控制到BackgroundWorker線程。

復制代碼 代碼如下:

BackgroundWorker bw = null;

 private void button1_Click(object sender, EventArgs e)
 {
     //創建一個BackgroundWorker線程
     bw = new BackgroundWorker();
     //指定可以讓線程停止
     bw.WorkerSupportsCancellation = true;
     //創建一個DoWork事件,指定bw_DoWork方法去做事
     bw.DoWork += new DoWorkEventHandler(bw_DoWork);
     //開始執行
     bw.RunWorkerAsync();
 }

 private void button2_Click(object sender, EventArgs e)
 {
     //停止線程
     bw.CancelAsync();
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     for (int i = 0; i 1000; i++)
     {
         //獲取當前線程是否得到停止的指令
         if (bw.CancellationPending)
         {
             e.Cancel = true;
             return;
         }

         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += i + Environment.NewLine;
         });
     }
 }

為了避免代碼的復雜化,上面代碼我沒有做更多的體驗修改,比如點擊開始的按鈕,開始的按鈕應該為不可用狀態,點擊停止按鈕后停止按鈕不可用狀態,激活開始按鈕。

下面我們將繼續升級,如何來獲知線程是否已經執行完成或者線程已經停止了呢

復制代碼 代碼如下:

BackgroundWorker bw = null;

 private void button1_Click(object sender, EventArgs e)
 {
     bw = new BackgroundWorker();
     bw.WorkerSupportsCancellation = true;
     bw.DoWork += new DoWorkEventHandler(bw_DoWork);
     //線程完成或者停止發生的事件
     bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);

     bw.RunWorkerAsync();
 }

 private void button2_Click(object sender, EventArgs e)
 {
     bw.CancelAsync();
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     for (int i = 0; i 1000; i++)
     {
         if (bw.CancellationPending)
         {
             e.Cancel = true;
             return;
         }

         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += i + Environment.NewLine;
         });
     }
 }

 void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
 {
     if (e.Cancelled)
     {
         this.richTextBox1.Text += "線程已經停止";
     }
     else
     {
         this.richTextBox1.Text += "線程已經完成";
     }
 }

到現在為止你可以自己去用BackgroundWorker創建一個線程了,你已經了解它了,當然BackgroundWorker還有一個ReportProgress滾動條事件,可以顯示進度,我暫且認為它是多余的,因為大部分進度都可以通過bw_DoWork來控制實現。下面我們繼續完善BackgroundWorker,加入暫停和繼續功能。

再拖動一個button控件到界面,BackgroundWorker的暫停和繼續我們使用ManualResetEvent。

復制代碼 代碼如下:

BackgroundWorker bw = null;
 //創建ManualResetEvent
 ManualResetEvent mr = new ManualResetEvent(true); 

 private void button1_Click(object sender, EventArgs e)
 {
     bw = new BackgroundWorker();
     bw.WorkerSupportsCancellation = true;
     bw.DoWork += new DoWorkEventHandler(bw_DoWork);
     bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);

     bw.RunWorkerAsync();
 }

 private void button2_Click(object sender, EventArgs e)
 {
     bw.CancelAsync();
 }

 private void button3_Click(object sender, EventArgs e)
 {
     Button b = (Button)sender;
     if (b.Text == "暫停")  
     {  
         mr.Reset();
         b.Text = "繼續";  
     }  
     else 
     {  
         mr.Set();  
         b.Text = "暫停";  
     } 

 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     for (int i = 0; i 1000; i++)
     {
         if (bw.CancellationPending)
         {
             e.Cancel = true;
             return;
         }

         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += i + Environment.NewLine;
         });

         //接受指令
         mr.WaitOne();
     }
 }

 void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
 {
     if (e.Cancelled)
     {
         this.richTextBox1.Text += "線程已經停止";
     }
     else
     {
         this.richTextBox1.Text += "線程已經完成";
     }
 }

到目前為止BackgroundWorker的大部分功能都實現了,上面的代碼在很多博客中都能找到,都是只執行了一個后臺線程。如果我們有1千個耗時的任務,那么一個線程遠遠不夠,我們需要創建多條線程,讓他分段執行,比如創建10個線程,把1千個任務分成不同的等分讓10個線程分別去執行。

我們使用list泛型 ListBackgroundWorker>,然后使用bw.RunWorkerAsync(i) 傳遞參數到bw_DoWork里,在bw_DoWork里使用e.Argument接受參數。

復制代碼 代碼如下:

ListBackgroundWorker> bws = new ListBackgroundWorker>();
 int t = 10;

 private void button1_Click(object sender, EventArgs e)
 {
     for (int i = 0; i t; i++)
     {
         BackgroundWorker bw = new BackgroundWorker();
         bw.DoWork += new DoWorkEventHandler(bw_DoWork);
         bws.Add(bw);

         bw.RunWorkerAsync(i);
     }
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     int j = Convert.ToInt32(e.Argument);
     for (int i = j; i 1000; i = i + t)
     {
         if (((BackgroundWorker)sender).CancellationPending) 
         {
             e.Cancel = true;
             return;
         }

         string item = String.Format("線程{0}正在操作數據{1}", j + 1, i);

         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += item + Environment.NewLine;
         });

         //Thread.Sleep(200);
     }
 }

由于上面代碼不是耗時操作,又開啟線程10個,操作過快,造成界面假死狀態,可以使用Sleep讓線程休眠。

我們繼續完善代碼,加入停止操作,加入完成后和停止的事件,由于是多線程,判斷是線程操作是否完成,我們用bws.Remove(sender as BackgroundWorker); 方法刪除線程,然后使用bws.Count == 0來判斷是否操作完成。

復制代碼 代碼如下:

ListBackgroundWorker> bws = new ListBackgroundWorker>();
 int t = 10;

 private void button1_Click(object sender, EventArgs e)
 {
     for (int i = 0; i t; i++)
     {
         BackgroundWorker bw = new BackgroundWorker();
         bw.DoWork += new DoWorkEventHandler(bw_DoWork);
         bw.WorkerSupportsCancellation = true;
         bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
         bws.Add(bw);

         bw.RunWorkerAsync(i);
     }
 }

 private void button2_Click(object sender, EventArgs e)
 {
     for (int i = 0; i t; i++)
     {
         bws[i].CancelAsync();
     }
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     int j = Convert.ToInt32(e.Argument);
     for (int i = j; i 1000; i = i + t)
     {
         if (((BackgroundWorker)sender).CancellationPending) 
         {
             e.Cancel = true;
             return;
         }

         string item = String.Format("線程{0}正在操作數據{1}", j + 1, i);

         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += item + Environment.NewLine;
         });

         Thread.Sleep(200);
     }
 }

 void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
 {
     bws.Remove(sender as BackgroundWorker);
     if (bws.Count == 0)
     {
         if (e.Cancelled)
         {
             this.richTextBox1.Text += "線程已經停止";
         }
         else
         {
             this.richTextBox1.Text += "線程已經完成";
         }
     }
 }

上面代碼中的停止不是能立即停止,這個就和開車一樣,開的越快,剎車的后拖行的距離越長,同理,開啟的線程的越多,完全暫停需要的時間越長,不知我說的是否正確。另外我也想問一下是否能真正的全部線程停止,點停止后全部線程立即停止。

下面我們繼續加入暫停和繼續的功能,一樣的道理,我們使用ListManualResetEvent>。

復制代碼 代碼如下:

ListBackgroundWorker> bws = new ListBackgroundWorker>();
 ListManualResetEvent> mrs = new ListManualResetEvent>(); 
 int t = 10;

 private void button1_Click(object sender, EventArgs e)
 {
     for (int i = 0; i t; i++)
     {
         BackgroundWorker bw = new BackgroundWorker();
         bw.DoWork += new DoWorkEventHandler(bw_DoWork);
         bw.WorkerSupportsCancellation = true;
         bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
         bws.Add(bw);

         bw.RunWorkerAsync(i);

         mrs.Add(new ManualResetEvent(true));
     }
 }

 private void button2_Click(object sender, EventArgs e)
 {
     for (int i = 0; i t; i++)
     {
         bws[i].CancelAsync();
     }
 }

 private void button3_Click(object sender, EventArgs e)
 {
     Button b = (Button)sender;  
     if (b.Text == "暫停")  
     {  
         for (int i = 0; i mrs.Count; i++)  
         {  
             mrs[i].Reset();  
         }  
         b.Text = "繼續";  
     }  
     else 
     {  
         for (int i = 0; i mrs.Count; i++)  
         {  
             mrs[i].Set();  
         }  
         b.Text = "暫停";  
     }  
 }

 void bw_DoWork(object sender, DoWorkEventArgs e)
 {
     int j = Convert.ToInt32(e.Argument);
     for (int i = j; i 1000; i = i + t)
     {
         if (((BackgroundWorker)sender).CancellationPending) 
         {
             e.Cancel = true;
             return;
         }

         string item = String.Format("線程{0}正在操作數據{1}", j + 1, i);

         this.Invoke((MethodInvoker)delegate
         {
             this.richTextBox1.Text += item + Environment.NewLine;
         });

         Thread.Sleep(200);
         mrs[j].WaitOne(); 
     }
 }

 void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
 {
     bws.Remove(sender as BackgroundWorker);
     if (bws.Count == 0)
     {
         if (e.Cancelled)
         {
             this.richTextBox1.Text += "線程已經停止";
         }
         else
         {
             this.richTextBox1.Text += "線程已經完成";
         }
     }
 }

至此,所有的代碼都奉上了,多個線程操作會帶來很多意向不到的麻煩,比如多個線程同時把數據寫入一個文件,多線程更新datatable等,會讓軟件莫名其妙的自動退出,.net2.0里還沒有絕對線程安全的數據集,很多大佬都說用lock,但我對lock也是一知半解,還請大家賜教賜教,如上有什么說的不對,也請大家多多指點。

您可能感興趣的文章:
  • c#異步操作后臺運行(backgroundworker類)示例
  • C#在后臺運行操作(BackgroundWorker用法)示例分享
  • C# BackgroundWorker組件學習入門介紹
  • asp.net BackgroundWorker之在后臺下載文件
  • C#實現winform用子窗體刷新父窗體及子窗體改變父窗體控件值的方法
  • C# WinForm程序處理后臺繁忙導致前臺控件假死現象解決方法
  • C#的winform控件命名規范
  • C#之WinForm跨線程訪問控件實例
  • WinForm的延時加載控件概述
  • WinForm中BackgroundWorker控件用法簡單實例

標簽:臺州 周口 朝陽 新鄉 百色 洛陽 朔州 喀什

巨人網絡通訊聲明:本文標題《簡單使用BackgroundWorker創建多個線程的教程》,本文關鍵詞  簡單,使用,BackgroundWorker,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《簡單使用BackgroundWorker創建多個線程的教程》相關的同類信息!
  • 本頁收集關于簡單使用BackgroundWorker創建多個線程的教程的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    欧美电影影音先锋| 欧美一级日韩一级| 欧美日韩日日摸| 久久这里只有精品视频网| 一区二区中文字幕在线| 久久99国产精品久久99果冻传媒| 成人午夜在线免费| 欧美成人精精品一区二区频| 亚洲码国产岛国毛片在线| 精品午夜一区二区三区在线观看| 日本道色综合久久| 国产欧美精品一区二区色综合朱莉 | 亚洲大尺度视频在线观看| 成人美女视频在线看| 日韩亚洲电影在线| 亚洲一区二区在线播放相泽| 成人黄色在线看| 久久久久久久久久美女| 日韩精品国产精品| 欧美午夜精品久久久久久孕妇| 国产精品灌醉下药二区| 国产成人午夜片在线观看高清观看| 91麻豆精品国产自产在线观看一区| 亚洲影院理伦片| 欧美在线啊v一区| 亚洲精品伦理在线| 91免费视频观看| 亚洲视频中文字幕| 色综合欧美在线| 亚洲欧美偷拍另类a∨色屁股| 99热精品国产| 中文字幕一区二区三区精华液| 国产精品91一区二区| 国产女人18毛片水真多成人如厕| 国产成人在线视频免费播放| 欧美精品一区二区精品网| 精品一区二区三区视频在线观看 | 成人免费视频caoporn| 欧美精品一区男女天堂| 久久精品免费看| 精品国产乱码久久久久久免费| 久久精品噜噜噜成人88aⅴ| 日韩精品一区二区三区中文不卡 | 久久久国产精华| 国产二区国产一区在线观看| 中文字幕精品一区二区三区精品| 成人av在线一区二区| 亚洲精品国产无天堂网2021| 欧美日免费三级在线| 日本系列欧美系列| 久久久久久免费网| caoporen国产精品视频| 一区二区三区在线免费视频| 777久久久精品| 国产一区二区三区在线观看免费视频| 欧美成人性战久久| 白白色 亚洲乱淫| 亚洲影视在线播放| 精品国产91亚洲一区二区三区婷婷| 国产乱色国产精品免费视频| 亚洲天堂av一区| 欧美乱妇15p| 成人性生交大合| 亚洲mv大片欧洲mv大片精品| 日韩欧美一级在线播放| 99久久久久免费精品国产 | 视频一区视频二区中文| www国产精品av| 色综合久久88色综合天天免费| 亚洲国产aⅴ成人精品无吗| 欧美成人aa大片| 成人的网站免费观看| 日韩激情一区二区| 国产精品久久久久久久久果冻传媒 | 国产欧美精品一区二区三区四区| 91原创在线视频| 黄色日韩网站视频| 一区二区三区电影在线播| 久久综合丝袜日本网| 欧美日韩一区二区三区高清| 国产精品夜夜爽| 午夜成人免费电影| 自拍偷拍亚洲综合| 国产日韩欧美高清在线| 91精品国产色综合久久ai换脸| 99亚偷拍自图区亚洲| 另类中文字幕网| 亚洲午夜激情av| 亚洲人成人一区二区在线观看| 精品国产一区二区在线观看| 欧美日本一区二区三区| 色综合久久综合网97色综合| 成人激情开心网| 狠狠色狠狠色合久久伊人| 五月婷婷综合激情| 亚洲在线成人精品| 亚洲精品亚洲人成人网在线播放| 久久久久久久久久久久久女国产乱 | 一区二区三区免费看视频| 亚洲国产精品高清| 久久久久国产免费免费| 欧美大黄免费观看| 在线播放/欧美激情| 欧美日韩一区二区欧美激情 | 91在线无精精品入口| 国产成人精品一区二区三区四区| 美女在线视频一区| 日韩高清在线不卡| 日本一道高清亚洲日美韩| 日韩中文字幕av电影| 午夜精品一区二区三区电影天堂| 亚洲激情在线激情| 自拍偷拍欧美精品| 一区二区三区在线观看欧美| 亚洲视频免费在线观看| 亚洲人成伊人成综合网小说| 亚洲婷婷在线视频| 亚洲成人免费在线| 视频一区中文字幕| 麻豆91在线播放免费| 紧缚奴在线一区二区三区| 久久99最新地址| 高清不卡一二三区| 91在线小视频| 欧美卡1卡2卡| 亚洲精品在线一区二区| 国产午夜精品久久久久久久| 国产日韩欧美精品综合| 中文字幕亚洲电影| 亚洲国产wwwccc36天堂| 久久激情五月激情| 国产69精品久久久久毛片| 91丨porny丨在线| 精品视频一区二区三区免费| 欧美日韩成人综合| 日韩久久久久久| 国产午夜精品福利| 亚洲午夜精品网| 精品制服美女丁香| 99国内精品久久| 欧美三级中文字幕在线观看| 日韩精品一区二区三区swag| 久久亚洲二区三区| 中文字幕佐山爱一区二区免费| 亚洲电影第三页| 国产综合成人久久大片91| 91在线观看污| 日韩欧美一级精品久久| 国产精品欧美久久久久无广告| 一区二区三区产品免费精品久久75| 久久国产综合精品| 91麻豆国产福利在线观看| 在线不卡中文字幕播放| 亚洲国产成人一区二区三区| 日韩电影免费在线| 97se亚洲国产综合自在线观| 欧美成人vps| 亚洲精品日产精品乱码不卡| 国内精品在线播放| 欧美午夜精品一区二区三区| 欧美激情综合网| 日韩制服丝袜先锋影音| av在线综合网| 国产网站一区二区三区| 日韩影视精彩在线| a4yy欧美一区二区三区| 精品日韩在线观看| 日日夜夜免费精品视频| 在线免费观看成人短视频| 国产欧美一区二区三区网站| 琪琪一区二区三区| 欧美日韩一区二区三区视频| 亚洲欧美激情小说另类| 国产一区二区毛片| 日韩视频一区二区| 日韩精品久久久久久| 欧美在线小视频| 亚洲精品久久7777| zzijzzij亚洲日本少妇熟睡| 久久久九九九九| 激情文学综合丁香| 337p日本欧洲亚洲大胆色噜噜| 午夜精品成人在线| 欧美精品自拍偷拍动漫精品| 一区二区三区高清| 日本久久电影网| 中文字幕亚洲区| av电影在线观看完整版一区二区| 久久精品视频一区| 国产黄色91视频| 久久九九久久九九| 国产99久久久久| 国产精品午夜在线观看| 国v精品久久久网| 国产欧美日产一区| 成人国产精品视频| 国产精品免费av| 99精品欧美一区二区三区小说| 亚洲视频 欧洲视频| 在线精品视频一区二区|