关于C#多线程委托的简单讲解
2014-11-14 09:20
363 查看
文章转载来自:http://lidunyang2008.blog.163.com/blog/static/19067620520132233112342/
Invoke()的作用是:在应用程序的主线程上执行指定的委托。一般应用:在辅助线程中修改UI线程( 主线程 )中对象的属性时,调用this.Invoke();
在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,Invoke
和 BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。
正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。
(参考:http://lidunyang2008.blog.163.com/blog/static/190676205201322325426803/)
而所谓的“一面响应操作,一面添加节点”永远只能是相对的,使 UI 线程的负担不至于太大而已,因为界面的正确更新始终要通过
UI 线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到
UI 线程中去做,这样也就达到了减轻 UI 线程负担的目的了。
(1)
//修改按钮的Enabled属性
private void ModifyButton( bool _b )
{
this.Button1.Enabled = _b;
}
(2)
//声明上面方法的委托
private delegate void ModifyButton_dg( bool _b );
(3)
//调用委托
private void Calldelgate( )
{
/*在Windows窗体应用程序中使用this.Invoke 在WPF应用程序中使用this.Dispatcher.Invoke*/
this.Invoke( new ModifyButton_dg( ModifyButton ) ,new object[]{false});
}
(4)
可以在非UI线程中调用 ,如:
//创建线程
Thread _t = new Thread( new ThreadStart( threadmethod ));
_t.Start();
//线程入口
private void threadmethod ()
{
//其他代码省略
Calldelgate();
}
线程会在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 Form1()
{
InitializeComponent();
}
①//更新窗体中的TextBox1显示内容的方法
public void UpdateForm(string param1,string
parm2)
{
this.textBox1.Text = param1+parm2;
}
②//声明委托
public delegate void UpdateForm_dl(string
str1, string str2);
③//调用委托
private void Calldelegate()
{
/*在Windows窗体应用程序中使用this.Invoke 在WPF应用程序中使用this.Dispatcher.Invoke*/
this.BeginInvoke(new UpdateForm_dl(UpdateForm), new object[] { "我是文本框", "haha" });
//this.Dispatcher.BeginInvoke(new UpdateForm_dl(UpdateForm), new object[] { "我是文本框", "haha" });
}
④//创建新线程
private void button1_Click(object sender, EventArgs
e)
{
Thread thread = new Thread(new
ThreadStart(DoWork));
thread.Start();
}
⑤//新线程入口
public void DoWork()
{
Calldelegate();
}
}
}
注意代理的使用!
Invoke()的作用是:在应用程序的主线程上执行指定的委托。一般应用:在辅助线程中修改UI线程( 主线程 )中对象的属性时,调用this.Invoke();
在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,Invoke
和 BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。
正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。
(参考:http://lidunyang2008.blog.163.com/blog/static/190676205201322325426803/)
而所谓的“一面响应操作,一面添加节点”永远只能是相对的,使 UI 线程的负担不至于太大而已,因为界面的正确更新始终要通过
UI 线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到
UI 线程中去做,这样也就达到了减轻 UI 线程负担的目的了。
this.invoke()用法:
(1)//修改按钮的Enabled属性
private void ModifyButton( bool _b )
{
this.Button1.Enabled = _b;
}
(2)
//声明上面方法的委托
private delegate void ModifyButton_dg( bool _b );
(3)
//调用委托
private void Calldelgate( )
{
/*在Windows窗体应用程序中使用this.Invoke 在WPF应用程序中使用this.Dispatcher.Invoke*/
this.Invoke( new ModifyButton_dg( ModifyButton ) ,new object[]{false});
}
(4)
可以在非UI线程中调用 ,如:
//创建线程
Thread _t = new Thread( new ThreadStart( threadmethod ));
_t.Start();
//线程入口
private void threadmethod ()
{
//其他代码省略
Calldelgate();
}
线程会在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 Form1()
{
InitializeComponent();
}
①//更新窗体中的TextBox1显示内容的方法
public void UpdateForm(string param1,string
parm2)
{
this.textBox1.Text = param1+parm2;
}
②//声明委托
public delegate void UpdateForm_dl(string
str1, string str2);
③//调用委托
private void Calldelegate()
{
/*在Windows窗体应用程序中使用this.Invoke 在WPF应用程序中使用this.Dispatcher.Invoke*/
this.BeginInvoke(new UpdateForm_dl(UpdateForm), new object[] { "我是文本框", "haha" });
//this.Dispatcher.BeginInvoke(new UpdateForm_dl(UpdateForm), new object[] { "我是文本框", "haha" });
}
④//创建新线程
private void button1_Click(object sender, EventArgs
e)
{
Thread thread = new Thread(new
ThreadStart(DoWork));
thread.Start();
}
⑤//新线程入口
public void DoWork()
{
Calldelegate();
}
}
}
注意代理的使用!
相关文章推荐
- 我关于c#中委托的简单理解
- C#实现异步编程的两个简单机制(异步委托&定时器)及Thread实现多线程
- 关于C#的委托入门讲解
- 我关于c#中委托的简单理解
- 【转载】C# 中的委托和事件(详解:简单易懂的讲解)
- 简单讲解c#委托delegate的用途
- 关于 C# 委托、事件
- C#中利用委托实现多线程跨线程操作
- .NET委托:一个关于C#的睡前故事
- 一个C#实现的最简单的委托例子
- 【原创】关于C#多线程安全问题的讨论
- MSDN中回调函数的讲解及其C#例子:用委托实现回调函数
- .NET委托:一个关于C#的睡前故事
- 2005-7-1关于C#中委托的使用
- 关于C#多播委托的若干问题
- .net2005(C#)下实现事件和委托的简单示例
- c# 委托与多线程
- .NET委托:一个关于C#的睡前故事
- C#中关于模式匹配的简单例子
- .NET委托:一个关于C#的睡前故事