C#中Invoke的用法

在做一个测试Tcp连接的DEMO,想用TcpClient.BeginConnect异步调用,在回调方法里更新winform.TextBox的值输出结果,直接使用TextBox.Text出错(线程间操作无效,不是从创建控件XX的线程访问他),想过使用winform.Invoke不过一直想Inovke将Tcp连接进行封装委托,原来应该将更新界面的方法进行委托。
最后代码
public partial class MainForm : Form
{
private string _serverAddress = string.Empty;
private int _port = 80;
delegate void SetTextCallback(string text);
public MainForm()
{
InitializeComponent();
}
private void connectButton_Click(object sender, EventArgs e)
{
if (string.IsNullOrEmpty(this.serverAddressTextBox.Text))
{
this.tipsTextBox.Text = "请输入服务器IP地址\r\n";
}
if (string.IsNullOrEmpty(this.portTextBox.Text))
{
this.tipsTextBox.Text="请输入端口号\r\n";
}
else
{
_serverAddress = this.serverAddressTextBox.Text;//连接IP地址
_port = Convert.ToInt32(this.portTextBox.Text);//端口号

this.tipsTextBox.Text = this.tipsTextBox.Text + "开始建立连接.....\r\n";
this.tipsTextBox.Text = this.tipsTextBox.Text + "请等待.....\r\n";
TcpClient tcpClient = new TcpClient();
//开启异步TCP连接
IAsyncResult asr=tcpClient.BeginConnect(_serverAddress, _port, new AsyncCallback(RequestCallBack), tcpClient);
//try
//{
// tcpClient.EndConnect(asr);
//}
//catch (System.Exception ex)
//{
// this.tipsTextBox.Text = this.tipsTextBox.Text + "错误:" + ex.Message + "\r\n";
//}
//finally
//{
// this.tipsTextBox.Text = this.tipsTextBox.Text + "操作是否完成:" + asr.IsCompleted + "\r\n";
// this.tipsTextBox.Text = this.tipsTextBox.Text + "连接结果:" + ((TcpClient)asr.AsyncState).Connected + "\r\n";
//}



}
}
/// <summary>
/// 异步回调函数
/// </summary>
/// <param name="asr"></param>
private void RequestCallBack(IAsyncResult asr)
{

TcpClient client = (TcpClient)asr.AsyncState;
SetText("连接结果:"+client.Connected.ToString());
if (client.Connected)
{
client.EndConnect(asr);
}
}
private void SetText(string text)
{
// InvokeRequired需要比较调用线程ID和创建线程ID
// 如果它们不相同则返回true
if (this.tipsTextBox.InvokeRequired)
{
this.Invoke(new SetTextCallback(SetText), new object[] { text });
}
else
{
this.tipsTextBox.Text = this.tipsTextBox.Text + "\n" + text;
}
}
private void MainForm_Load(object sender, EventArgs e)
{
// Control.CheckForIllegalCrossThreadCalls = false;
}
}

在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,Invoke 和 BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。

正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。

而所谓的“一面响应操作,一面添加节点”永远只能是相对的,使 UI 线程的负担不至于太大而已,因为界面的正确更新始终要通过 UI 线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到 UI 线程中去做,这样也就达到了减轻 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 delegate void MyInvoke(string str1,string str2);
public Form1()
{
InitializeComponent();


}
public void DoWork()
{
MyInvoke mi = new MyInvoke(UpdateForm);
this.BeginInvoke(mi, new Object[] {"我是文本框","haha"});
}
public void UpdateForm(string param1,string parm2)
{
this.textBox1.Text = param1+parm2;
}
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(DoWork));
thread.Start();
}
}
}
注意代理的使用!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值