c#如何跨线程调用窗体控件
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了c#如何跨线程调用窗体控件相关的知识,希望对你有一定的参考价值。
要从其他跨线程存取调用控件,可采用以下两种方法之一:
方法1)不进行线程安全的检查
方法2)通过委托的方式
代码如下所示
public Form1()
InitializeComponent();
//方法1:不进行跨线程安全检查
//System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;
private void button1_Click(object sender, EventArgs e)
Thread th1 = new Thread(new ThreadStart(CalNum));
th1.Start();
private void CalNum()
//button1.Enabled = false;
int result = 0;
for (int i = 1; i < 100000000; i++)
result += i;
SetCalResult(result);
//button1.Enabled = true;
//方法2:检查是否跨线程,然后将方法加入委托,调用委托
public delegate void SetTextHandler(int result);
private void SetCalResult(int result)
if (label2.InvokeRequired == true)
SetTextHandler set = new SetTextHandler(SetCalResult);//委托的方法参数应和SetCalResult一致
label2.Invoke(set, new object[] result ); //此方法第二参数用于传入方法,代替形参result
else
label2.Text = result.ToString();
参考技术A 如果你面临无数的控件和控件组,我看你怎么玩下去。所以在C#中我们经常讲的线程安全,意义就在这里,你要学会把这些东西缓存在内存中,而不是让程序去不断的invoke才是程序的关键所在。你不要天真的以为invoke一定是安全的,也不要天真的认为invoke不是事件,他也在运行。在使用多线程的时候我们完全可以把变量定义为全局变量。android studio这一点就比C#机制做得好,不会这么紊乱至少。 参考技术B Invoke方法吧,不进行线程安全的检查不好 参考技术C new Action(delegate
ListBox.Invoke(new Action(delegate
ListBox.Item.Add("hello!")
));
).BingInvoke(null,null);
c#中如何跨线程调用windows控件
参考技术A 在辅助线程调用(在创建辅助线程时可将此方法通过delegate传到辅助线程中)下面的方法InvokeControl();//写在主线程中(windows控件)
private void InvokeControl()
if (this.InvokeRequired)
this.Invoke(new DelegateChangeText(ChangeText));
else
this.ChangeText();
private void ChangeText()
this.TextBox.Text = "sd";
public delegate void DelegateChangeText(); 参考技术B 定义委托,然后用invoke
private
void
ucMidPartsid_Load(object
sender,
EventArgs
e)
if
(is_Load
==
true)
thread
=
new
Thread(new
ThreadStart(this.LoadData));
thread.Start();
//if
(datafinish
!=
null)
//
//
datafinish(this,
e);
//
private
void
LoadData()
QueryDataCallBack
deleg;
deleg
=
new
QueryDataCallBack(this.DataBinding);
this.Invoke(deleg,null);
private
void
DataBinding()
partstableAdapter
=
new
GreatWall.SCM.Parts.DataAccess.DAParts.PartsTableAdapter();
partstableAdapter.Fill(dsParts.GB_PARTS);
cobMidPartsid.DataSource
=
dsParts.GB_PARTS;
cobMidPartsid.DisplayMember
=
"B_PARTSID";
cobMidPartsid.ValueMember
=
"B_ID";
EventArgs
e
=
new
EventArgs();
if
(datafinish
!=
null)
finish
=
true;
datafinish(this,
e);
参考技术C 在构造函数里使用属性
CheckForIllegalCrossThreadCalls 值设置为 false
即 public Form1()
CheckForIllegalCrossThreadCalls = false;
参考技术D 定义委托,然后用invoke
private void ucMidPartsid_Load(object sender, EventArgs e)
if (is_Load == true)
thread = new Thread(new ThreadStart(this.LoadData));
thread.Start();
//if (datafinish != null)
//
// datafinish(this, e);
//
private void LoadData()
QueryDataCallBack deleg;
deleg = new QueryDataCallBack(this.DataBinding);
this.Invoke(deleg,null);
private void DataBinding()
partstableAdapter = new GreatWall.SCM.Parts.DataAccess.DAParts.PartsTableAdapter();
partstableAdapter.Fill(dsParts.GB_PARTS);
cobMidPartsid.DataSource = dsParts.GB_PARTS;
cobMidPartsid.DisplayMember = "B_PARTSID";
cobMidPartsid.ValueMember = "B_ID";
EventArgs e = new EventArgs();
if (datafinish != null)
finish = true;
datafinish(this, e);
本回答被提问者采纳 第5个回答 2007-12-19 定义委托 delegate
调用 Invoke
提问之前先利用baidu搜索下吧
以上是关于c#如何跨线程调用窗体控件的主要内容,如果未能解决你的问题,请参考以下文章
C# Winform项目中多线程环境下, 如何跨线程对Window窗体控件进行安全访问?