系统跨线程调用控件

窗体上放了一个listBox1,创建了两个线程,我再线程中访问这个listBox1的时候,系统说不能跨线程调用控件,这怎么回事 ,怎么解决??
代码如下:
private void Form1_Load(object sender, EventArgs e)
{
//创建线程 需要System.Threading命名空间
Thread t1, t2; // 说明为窗体类成员
t1 = new Thread(new ThreadStart(BackgroundProcess));
t1.Start(); //启动线程t1
}
private void BackgroundProcess()
{
int i = 1;
while (true)
{
// 向列表框增加一个项目
listBox1.Items.Add("Iterations: " + i.ToString());


i++;
Thread.Sleep(2000); // 指定线程休眠的时间
}


 
      如果代码是在VS2003中的话应该不会抛出这个异常,只有在使用VS2005的时候,如果你从非创建这个控件的线程中访问这个控件或者操作这个控件的话就会抛出这个异常。这是微软为了保证线程安全以及提高代码的效率所做的改进.

以前在VS2003中,我们通常都会忽略这种问题的存在,无限制的使用线程来完成一些工作,但是当系统慢慢变大的时候,这种滥用的线程不但没有提高我们的效率,反而使我们找错误的效率以及系统的复杂度大大的增加了。

微软的这一改动可以让我们在使用线程的时候,充分考虑一下线程的安全以及线程的使用规范,这样会有益于我们做出健壮的程序。

坚决这种跨线程的问题在VS2005中提供了两种简单的方法(我个人当前所了解的只有这两种):
一种是通过设置System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;(winform下)如果在你的程序初始化的时候设置了这个属性,而且在你的控件中使用的都是微软Framework类库中的控件的话,系统就不会再抛出你上面所说的这个错误了。当然这只是为了将VS2003的代码转换到VS2005下所使用的一种常见的方法。不建议采用。

第二种方法就是微软建议采用的跨线程调用的一种通用方法,就是使用代理来实现,就是将你所要操作的代码放到一个代理种,然后将这个代理交给创建这个控件的线程来执行你的代码。例如:
private void Form1_Load(object sender, EventArgs e)
{
//创建线程 需要System.Threading命名空间
Thread t1, t2; // 说明为窗体类成员
t1 = new Thread(new ThreadStart(BackgroundProcess));
t1.Start(); //启动线程t1
}
/// <summary>
/// 定义一个代理
/// </summary>
private delegate void dd();

private void BackgroundProcess()
{
// 将代理实例化为一个匿名代理
dd = delegate()
{
int i = 1;
while (true)
{
// 向列表框增加一个项目
listBox1.Items.Add("Iterations: " + i.ToString());


i++;
Thread.Sleep(2000); // 指定线程休眠的时间
}
};
listBox1.Invoke(dd);
}
上面这个代码只是在你的代码中声明了一个代理,并且用VS2005中新加的语法(匿名代理,又名匿名方法。)来实例话这个代理,将你在线程中要操作的代码都放到这个匿名的方法中去。然后通过使用控件的Invoke方法(也可以使用控件的BeginInvoke方法——Invoke方法是同步的BeginInvoke方法是异步的)来调用这个代理。当然在Invoke方法中你也可以输入任何形式的代理。通过这种方式调用的代码就不会抛出你出现的那个异常了。

上面这种代理的方式类似于C++中的回调函数,你写出了执行的方法,然后通知某个线程由那个线程来调用你的这个方法,这样就做到了在固定的线程里执行修改线程内部组件的方式。这样就完全达到线程安全了。

另外,如果你使用的不是.NET类库中的控件的话,最好不要使用第一种方式,因为某些第三方控件在线程安全性方面做的不是太好,可能还会抛出一些不可预料的异常。例如DevExpress控件库一般会抛出对象引用为空的异常。所以建议所有在VS2005种使用线程操作控件的时候都利用第二种方法来实现,如果嫌麻烦,也可以将这些控件利用继承的方式在父类里面实现这些代理方式的修改。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
异步委托线程调用控件是指在多线程的情况下,通过使用异步委托来实现对控件线程调用。这种方法可以解决在主线程中访问控件时可能出现的界面卡死、无响应等问题。 在C#中,可以通过使用Invoke或BeginInvoke方法来实现异步委托线程调用控件。使用Invoke方法可以确保任务在主线程上执行,而BeginInvoke方法允许任务在后台线程上执行。具体的使用方法如下: 1. 在主线程中创建委托方法,该方法用于更新控件的属性或执行其他与控件相关的操作。 2. 当需要在后台线程中更新控件时,使用Invoke或BeginInvoke方法,将委托方法作为参数传递给它们。 3. 在委托方法中,通过控件的Invoke方法或BeginInvoke方法来更新控件。 请注意,为了避免线程冲突和不可预料的结果,强烈不推荐使用Control.CheckForIllegalCrossThreadCalls = false;这种方法。相反,应该采用合理的方法,即在主线程中负责UI界面的调度,将耗时的操作放在后台线程中进行,并使用Invoke委托来触发UI刷新。这样可以确保界面的稳定性和响应性,避免出现卡死、无响应或报错等问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [C#线程调用控件](https://blog.csdn.net/tonghaojie/article/details/78767046)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [C# winform delegate委托以及异步调用委托(线程调用控件UI)使用总结](https://blog.csdn.net/gxlzhhx/article/details/128568462)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值