当我运行以下运行正常但未按预期运行的代码时,假设它在安全线程上运行,但所有组件都被冻结,直到它完成运行线程,它不应该在新线程上运行所以你可以使用其他控件吗?
using System;
using System.Drawing;
using System.Windows.Forms;
using System.Threading;
public class MyFormControl : Form
{
public delegate void AddListItem();
public AddListItem myDelegate;
private Button myButton;
private Thread myThread;
private ListBox myListBox;
public MyFormControl()
{
myButton = new Button();
myListBox = new ListBox();
myButton.Location = new Point(72, 160);
myButton.Size = new Size(152, 32);
myButton.TabIndex = 1;
myButton.Text = "Add items in list box";
myButton.Click += new EventHandler(Button_Click);
myListBox.Location = new Point(48, 32);
myListBox.Name = "myListBox";
myListBox.Size = new Size(200, 95);
myListBox.TabIndex = 2;
ClientSize = new Size(292, 273);
Controls.AddRange(new Control[] {myListBox,myButton});
Text = " 'Control_Invoke' example";
myDelegate = new AddListItem(AddListItemMethod);
}
static void Main()
{
MyFormControl myForm = new MyFormControl();
myForm.ShowDialog();
}
public void AddListItemMethod()
{
String myItem;
for(int i=1;i<6;i++)
{
myItem = "MyListItem" + i.ToString();
myListBox.Items.Add(myItem);
myListBox.Update();
Thread.Sleep(300);
}
}
private void Button_Click(object sender, EventArgs e)
{
myThread = new Thread(new ThreadStart(ThreadFunction));
myThread.Start();
}
private void ThreadFunction()
{
MyThreadClass myThreadClassObject = new MyThreadClass(this);
myThreadClassObject.Run();
}
}
// The following code assumes a 'ListBox' and a 'Button' control are added to a form,
// containing a delegate which encapsulates a method that adds items to the listbox.
public class MyThreadClass
{
MyFormControl myFormControl1;
public MyThreadClass(MyFormControl myForm)
{
myFormControl1 = myForm;
}
public void Run()
{
// Execute the specified delegate on the thread that owns
// 'myFormControl1' control's underlying window handle.
myFormControl1.Invoke(myFormControl1.myDelegate);
}
}
答案 0 :(得分:2)
假设它在安全线程上运行,但所有组件都被冻结 直到它完成运行线程
当您在控件上调用某个委托时,该委托将在UI线程上运行。即此代码将在UI线程上运行:
public void AddListItemMethod()
{
String myItem;
for(int i=1;i<6;i++)
{
myItem = "MyListItem" + i.ToString();
myListBox.Items.Add(myItem);
myListBox.Update();
Thread.Sleep(300); // freeze UI thread
}
}
它不应该在新线程上运行,因此您可以使用其他线程 控制?
您无法使用非UI线程中的控件。
使用后台线程的目的是长期运行的操作,与UI无关。例如。你可以从磁盘读取一些文件,查询api,或者你可以运行一些长时间运行的计算(第n个斐波纳契数)。如果你在UI线程上运行这些东西,那么你的应用程序将会冻结。因此,您应该在非UI线程上运行此类操作,并在完成长时间运行后返回UI (尽管您可以通知用户有关长时间运行操作的进度)。
如果您想定期对UI执行某些操作,请考虑使用System.Windows.Forms.Timer
组件。将计时器间隔设置为300并添加Tick事件处理程序:
private void Button_Click(object sender, EventArgs e)
{
timer.Start();
}
private void Timer_Tick(object sender, EventArgs e)
{
myListBox.Items.Add($"MyListItem{myListBox.Items.Count + 1}");
}
答案 1 :(得分:0)
问题是Invoke
在UI线程上运行委托,所以你只是创建一个只告诉UI线程完成所有工作的线程。相反,您可以使用async
和await
以及Task.Delay
来简化代码。
private async void Button_Click(object sender, EventArgs e)
{
String myItem;
for(int i=1;i<6;i++)
{
myItem = "MyListItem" + i.ToString();
myListBox.Items.Add(myItem);
myListBox.Update();
await Task.Delay(300);
}
}
这将在UI上运行代码,但现在await Task.Delay(300)
将不会阻止UI线程,从而允许它处理其他UI事件并停止冻结问题。