net2.0以后,微软加强了控件的安全机制,不允许跨线程操作。
如果想跨线程操作控件的话要使用委托。
以前有过相关学习,后来既不清了,看了一遍之前写的文章,觉得太烂了,重新总结一下:
思路:在主线程中定义一个方法,这个方法用于主线程,然后定义一个委托,将委托指向这个方法(或者说讲方法绑定到委托上),其他线程调用委托。达到迂回的调用了主线程的方法,控制主线程的资源。
ps:这里之所以用“委托”这个词,不用“代理”,主要是感觉应用环境不同。线程A操作线程B的资源,但是不允许直接操作,那么A只好访问一个允许访问的对象,然后在委托这个对象对B的资源进行操作。站在这个角度“委托”更合适一点。
知识点:
Invoke 、InvokeRequired、delegate
新建一个winform工程,添加一个按钮、一个进度条名字默认: button1、progressBar1
例子一,无参数无返回值的方法:
1、添加一个代理:
private delegate void myDelegate1();
2、为修改进度条创建一个方法:
private void SetProgressBar1()
{
progressBar1.Value = 0;
progressBar1.Maximum = 2;
this.progressBar1.Value = this.progressBar1.Value + 1;
MessageBox.Show("SetProgressBar1");
}
3、添加一个线程将要调用的方法:
private void TRun1()
{
//判断控件是否在本线程内
if (!this.progressBar1.InvokeRequired)
{
// MessageBox.Show("同一线程内");
Thread.Sleep(2000);
}
else
{
myDelegate1 md1 = new myDelegate1(SetProgressBar1);
//MessageBox.Show("不是同一个线程");
Thread.Sleep(2000);
Invoke(md1);
}
}
4、开启新线程调用:
private void button1_Click(object sender, EventArgs e)
{
// RunWithInvoke();
Thread thread = new Thread(new ThreadStart(TRun1));
thread.Start();
MessageBox.Show("使用多线程");
}
结果:单击button1之后弹出提示框“使用多线程”,两秒钟之后弹出提示框“SetProgressBar1”并且进度条增长到50%位置。
执行过程:
1、先创建一个新线程并指向“TRun1”方法
2、“TRun1”方法判断控件是否属于该线程
3、不属于该线程,创建一个代理,并将代理指向“SetProgressBar1”方法
4、停顿2秒钟,Invoke代理,从而执行“SetProgressBar1”方法。(“SetProgressBar1”方法在主线程地址空间执行)
例子二,有参数的方法:
1、 创建代理
private delegate void myDelegate2(string s);
2、创建方法
//跟SetProgressBarValue委托相匹配的方法
private void SetProgressBar2(string s)
{
progressBar1.Value = 0;
progressBar1.Maximum = 2;
this.progressBar1.Value = this.progressBar1.Value + 1;
MessageBox.Show("SetProgressBar2 "+s);
}
3、 添加线程的方法
private void TRun2()
{
//判断控件是否在本线程内
if (!this.progressBar1.InvokeRequired)
{
// MessageBox.Show("同一线程内");
Thread.Sleep(2000);
}
else
{
myDelegate2 md2 = new myDelegate2(SetProgressBar2);
//MessageBox.Show("不是同一个线程");
Thread.Sleep(2000);
Invoke(md2, "TRun2");
}
}
4、 新线程
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(TRun2));
thread.Start();
MessageBox.Show("使用多线程");
}
例子三,有返回值的方法
1 、创建代理
private delegate string myDelegate3(string s);
2、创建方法
private string SetProgressBar3(string s)
{
progressBar1.Value = 0;
progressBar1.Maximum = 2;
this.progressBar1.Value = this.progressBar1.Value + 1;
MessageBox.Show("SetProgressBar3 " + s);
return ("return SetProgressBar3 " + s);
}
3、添加线程的方法
private void TRun3()
{
//判断控件是否在本线程内
if (!this.progressBar1.InvokeRequired)
{
// MessageBox.Show("同一线程内");
Thread.Sleep(2000);
}
else
{
myDelegate3 md3 = new myDelegate3(SetProgressBar3);
//MessageBox.Show("不是同一个线程");
Thread.Sleep(2000);
MessageBox.Show ( Invoke(md3, "TRun3").ToString());
}
}
4、 新线程
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(TRun3));
thread.Start();
MessageBox.Show("使用多线程");
}
以前有过相关学习,后来既不清了,看了一遍之前写的文章,觉得太烂了,重新总结一下:
思路:在主线程中定义一个方法,这个方法用于主线程,然后定义一个委托,将委托指向这个方法(或者说讲方法绑定到委托上),其他线程调用委托。达到迂回的调用了主线程的方法,控制主线程的资源。
ps:这里之所以用“委托”这个词,不用“代理”,主要是感觉应用环境不同。线程A操作线程B的资源,但是不允许直接操作,那么A只好访问一个允许访问的对象,然后在委托这个对象对B的资源进行操作。站在这个角度“委托”更合适一点。
知识点:
新建一个winform工程,添加一个按钮、一个进度条名字默认: button1、progressBar1
1、添加一个代理:
private delegate void myDelegate1();
2、为修改进度条创建一个方法:
private void SetProgressBar1()
{
progressBar1.Value = 0;
progressBar1.Maximum = 2;
this.progressBar1.Value = this.progressBar1.Value + 1;
MessageBox.Show("SetProgressBar1");
}
3、添加一个线程将要调用的方法:
private void TRun1()
{
//判断控件是否在本线程内
if (!this.progressBar1.InvokeRequired)
{
// MessageBox.Show("同一线程内");
Thread.Sleep(2000);
}
else
{
myDelegate1 md1 = new myDelegate1(SetProgressBar1);
//MessageBox.Show("不是同一个线程");
Thread.Sleep(2000);
Invoke(md1);
}
}
4、开启新线程调用:
private void button1_Click(object sender, EventArgs e)
{
// RunWithInvoke();
Thread thread = new Thread(new ThreadStart(TRun1));
thread.Start();
MessageBox.Show("使用多线程");
}
结果:单击button1之后弹出提示框“使用多线程”,两秒钟之后弹出提示框“SetProgressBar1”并且进度条增长到50%位置。
执行过程:
1、先创建一个新线程并指向“TRun1”方法
2、“TRun1”方法判断控件是否属于该线程
3、不属于该线程,创建一个代理,并将代理指向“SetProgressBar1”方法
4、停顿2秒钟,Invoke代理,从而执行“SetProgressBar1”方法。(“SetProgressBar1”方法在主线程地址空间执行)
1、
private delegate void myDelegate2(string s);
2、创建方法
//跟SetProgressBarValue委托相匹配的方法
private void SetProgressBar2(string s)
{
progressBar1.Value = 0;
progressBar1.Maximum = 2;
this.progressBar1.Value = this.progressBar1.Value + 1;
MessageBox.Show("SetProgressBar2 "+s);
}
3、 添加线程的方法
private void TRun2()
{
//判断控件是否在本线程内
if (!this.progressBar1.InvokeRequired)
{
// MessageBox.Show("同一线程内");
Thread.Sleep(2000);
}
else
{
myDelegate2 md2 = new myDelegate2(SetProgressBar2);
//MessageBox.Show("不是同一个线程");
Thread.Sleep(2000);
Invoke(md2, "TRun2");
}
}
4、 新线程
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(TRun2));
thread.Start();
MessageBox.Show("使用多线程");
}
1
private delegate string myDelegate3(string s);
2、创建方法
private string SetProgressBar3(string s)
{
progressBar1.Value = 0;
progressBar1.Maximum = 2;
this.progressBar1.Value = this.progressBar1.Value + 1;
MessageBox.Show("SetProgressBar3 " + s);
return ("return SetProgressBar3 " + s);
}
3、添加线程的方法
private void TRun3()
{
//判断控件是否在本线程内
if (!this.progressBar1.InvokeRequired)
{
// MessageBox.Show("同一线程内");
Thread.Sleep(2000);
}
else
{
myDelegate3 md3 = new myDelegate3(SetProgressBar3);
//MessageBox.Show("不是同一个线程");
Thread.Sleep(2000);
MessageBox.Show ( Invoke(md3, "TRun3").ToString());
}
}
4、 新线程
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(TRun3));
thread.Start();
MessageBox.Show("使用多线程");
}