转 - 跨线程访问
可以有两中方法:
一种是通过设置System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;(winform下)如果在你的程序初始化的时候设置了这个属性,而且在你的控件中使用的都是微软Framework类库中的控件的话,系统就不会再抛出你上面所说的这个错误了。当然这只是为了将VS2003的代码转换到VS2005下所使用的一种常见的方法。不建议采用。
第二种方法就是微软建议采用的跨线程调用的一种通用方法,就是使用代理来实现,就是将你所要操作的代码放到一个代理种,然后将这个代理交给创建这个控件的线程来执行你的代码。例如:
1 private void Form1_Load(object sender, EventArgs e) 2 { 3 //创建线程 需要System.Threading命名空间 4 Thread t1, t2; // 说明为窗体类成员 5 t1 = new Thread(new ThreadStart(BackgroundProcess)); 6 t1.Start(); //启动线程t1 7 } 8 /// <summary> 9 /// 定义一个代理 10 /// </summary> 11 private delegate void dd(); 12 13 private void BackgroundProcess() 14 { 15 // 将代理实例化为一个匿名代理 16 dd = delegate() 17 { 18 int i = 1; 19 while (true) 20 { 21 // 向列表框增加一个项目 22 listBox1.Items.Add("Iterations: " + i.ToString()); 23 24 25 i++; 26 Thread.Sleep(2000); // 指定线程休眠的时间 27 } 28 }; 29 listBox1.Invoke(dd); 30 }
上面这个代码只是在你的代码中声明了一个代理,并且用VS2005中新加的语法(匿名代理,又名匿名方法。)来实例话这个代理,将你在线程中要操作的代码都放到这个匿名的方法中去。然后通过使用控件的Invoke方法(也可以使用控件的BeginInvoke方法——Invoke方法是同步的BeginInvoke方法是异步的)来调用这个代理。当然在Invoke方法中你也可以输入任何形式的代理。通过这种方式调用的代码就不会抛出你出现的那个异常了。
上面这种代理的方式类似于C++中的回调函数,你写出了执行的方法,然后通知某个线程由那个线程来调用你的这个方法,这样就做到了在固定的线程里执行修改线程内部组件的方式。这样就完全达到线程安全了。
------------------------------------------------------------------------------------------------------------------------------------------------
1 try.. 2 3 //定义一个互斥量,用于对ListBox的互斥访问 4 Mutex mx=new Mutex(); 5 public delegate void MyInvoke(string str); 6 private void UpdateListBox(string str) 7 { 8 //更新 9 mx.WaitOne(); 10 this.listBox1.Items.Add(str); 11 mx.ReleaseMutex(); 12 } 13 private void button1_Click(object sender, EventArgs e) 14 { 15 Thread OneThread = new Thread(new ThreadStart(this.Run1)); 16 OneThread.Name = "one"; 17 Thread TowThread = new Thread(new ThreadStart(this.Run2)); 18 TowThread.Name = "tow"; 19 OneThread.Start(); 20 TowThread.Start(); 21 } 22 private void Run1() 23 { 24 for (int i = 0; i < 100; i++) 25 { 26 27 MyInvoke mi=new MyInvoke(UpdateListBox); 28 this.BeginInvoke(mi,new object[]{Thread.CurrentThread.Name + i.ToString()}); 29 } 30 } 31 private void Run2() 32 { 33 for (int i = 0; i > -100; i--) 34 { 35 MyInvoke mi=new MyInvoke(UpdateListBox); 36 this.BeginInvoke(mi,new object[]{Thread.CurrentThread.Name + i.ToString()}); 37 } 38 }