线程安全C#无法正常工作

时间:2017-04-06 12:38:28

标签: c# multithreading

当我运行以下运行正常但未按预期运行的代码时,假设它在安全线程上运行,但所有组件都被冻结,直到它完成运行线程,它不应该在新线程上运行所以你可以使用其他控件吗?

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);
      }
   }

2 个答案:

答案 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线程完成所有工作的线程。相反,您可以使用asyncawait以及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事件并停止冻结问题。