c#如何跨线程调用窗体控件

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了c#如何跨线程调用窗体控件相关的知识,希望对你有一定的参考价值。

要从其他跨线程存取调用控件,可采用以下两种方法之一:

方法1)不进行线程安全的检查

方法2)通过委托的方式

代码如下所示

public partial class Form1 : Form
    
        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#如何跨线程调用窗体控件的主要内容,如果未能解决你的问题,请参考以下文章

如何跨线程调用Windows窗体控件

C# Winform项目中多线程环境下, 如何跨线程对Window窗体控件进行安全访问?

C# 如何跨线程对Window窗体控件进行安全访问 (第二版)?

VB跨线程调用WPF控件

C# winform 跨线程更改窗体控件的属性

C# winform 跨线程操作winform程序窗体