C# 多线程编程

C# 多线程编程1.如果只是启动一个新线程,不需要传入参数,不需要线程返回结果,可以直接使用ThreadStart(),Thread类接收一个ThreadStart委托或ParameterizedThreadSta

大家好,又见面了,我是你们的朋友全栈君。

1.如果只是启动一个新线程,不需要传入参数,不需要线程返回结果,可以直接使用ThreadStart(),

Thread类接收一个ThreadStart委托或ParameterizedThreadStart委托的构造函数。ThreadStart委托中  作为参数的方法   不需要参数,并且没有返回值。ParameterizedThreadStart委托,可以在线程start的时候,传入参数,利用这个参数可以向线程传递参数(ParameterizedThreadStart是一个有参的、返回值为void的委托,参数类型必须是object)。

private void btOnlineLogin_Click(object sender, EventArgs e) 
    {
        Thread CheckStatusThread = new Thread(new ThreadStart(CheckStatus));       
     CheckStatusThread.Start(); //无参和返回值的多线程 
  }

private void CheckStatus()
{
  //do some thing here
}

2. 如果需要向线程传入参数,但不需要线程返回值,用ParameterizedThreadStart是一个有参的、返回值为void的委托,参数类型必须是object,且只能有一个object参数传入,
如果要传入多个参数,可以用下面3中的Func()或者定义一个结构体,结构体包含你要传入的参数,然后把结构体当作object传入。

private void btOnlineLogin_Click(object sender, EventArgs e) 
{
        Thread CheckStatusThread = new Thread(new ParameterizedThreadStart(CheckStatus));       
    CheckStatusThread.Start("print me"); //有传入参,无返回值的多线程 
}

private void CheckStatus(object value)
{
  Console.WriteLine(value)
}
//==============================总之上述用Thread方式,都没有返回值。

3.如果新线程不需要传入参数,但是需要得到返回结果,可以有两种做法:

a: 定义一个全局变量,按照上述1中方法启动一个无参无函数返回值的线程,将要返回的结果设置给全局变量

  public class Class1
  {
    bool Ready= false;
    private void btOnlineLogin_Click(object sender, EventArgs e)
    {
      Thread CheckStatusThread = new Thread(new ThreadStart(CheckStatus));
      CheckStatusThread.Start(); //无参和返回值的多线程

      while(!Ready)
      {
        Console.WriteLine(“wait for status to ready”);
      }

    }

    private void CheckStationLockedWindow()
    {
      System.Threading.Thread.Sleep(1000); //使 UI 线程有了处理界面消息的机会
      Ready = true;
    }

  }

 

b.使用Func(in_param_type, in_param_type, out_param_type), 然后使用Func的BeginInvoke() 异步启动一个新线程,该做法界面会卡住。

private void button33_Click(object sender, EventArgs e)
{
  Func<bool> test = MyTestMethod;
  // Func<bool> test = new Func<bool>(MyTestMethod); //这样一样
  IAsyncResult result = test.BeginInvoke(null, null); //不用传入参数
  bool fin = test.EndInvoke(result);
  if(fin)
  {
    Console.WriteLine(“OK………”);
  }

}

private bool WaitAndCreateEmptyFwFolder()
{
  Stopwatch timer = new Stopwatch();
  timer.Reset();
  timer.Start();
  while(timer.ElapsedMilliseconds <20*1000)
  {

    System.Threading.Thread.Sleep(1000); //使 UI 线程有处理界面消息的机会,防止卡住

    Application.DoEvents();

    Console.WriteLine(“Elappsed time:” + timer.ElapsedMilliseconds.ToString());
  }
  return true;
}

3. 如果即需要给线程传入参数还需要得到线程返回值,还是可以用Func(), 用Func() 可以不用定义自己的delegate了。

public class Class1
{ 
   private void button33_Click(object sender, EventArgs e)
  {
    Func<string, bool> test = PrintSomeThing;
    IAsyncResult result = test.BeginInvoke(“Print me”, null, null);
    bool fin = test.EndInvoke(result);

if (fin)
{
Console.WriteLine(“OK………”);
}

  }

  private bool PrintSomeThing(string arg)

  {

Stopwatch timer = new Stopwatch();
timer.Reset();
timer.Start();
while (timer.ElapsedMilliseconds < 10 * 1000)
{

  System.Threading.Thread.Sleep(1000); //使 UI 线程有处理界面消息的机会

    Application.DoEvents();
    Console.WriteLine(arg);
  }
return true;

}

}

 

4. 第四种多线程用task:System.Thread.Tasks.Task

Task的参数有两种,一种是Action(无返回值),一种是Func。

public class Class1

{

   private void button33_Click(object sender, EventArgs e)

     {

//==============带有返回值和传入参数的
//Task task  = Task<bool>.Factory.StartNew(new Func<object,bool>(TaskMthod),"321");   //带有返回值和传入参数的
Task<bool> task = new Task<bool>(TaskMthodWithReturnAndInParam, "123");   //new Task<>()中的<>里面放返回值类型,
                                     //()里面放Func类型method名字。
                                     // 如果直接用new Task()没有<>则没有返回值,里面放Action类型method名字。

task.Start();//start async          

//Console.WriteLine(task.Result.ToString()); //task.Result will block the thread unitil the task is done and run to next step, UI will stuck
//task.Wait(); //will block until done
    while(!task.IsCompleted) // this won't block UI
          {
                Application.DoEvents();
          }
          Console.WriteLine(task.Result.ToString());
//================只有返回值,无传入参数的
Task<bool> task2 = new Task<bool>(TaskMthodWithReturnNoInParam);          
task2.Start();//start async
   while(!task2.IsCompleted)
          {
                Application.DoEvents();
          }
Console.WriteLine(task2.Result.ToString());
//Task.WaitAll(task1,task2); //wait all task done

//===================没有返回值,只有传入参数的
Task task = Task.Run(() => TaskMthodWithInParamOnly(true, true));          
while (!task.IsCompleted)          
{              
Application.DoEvents();          
}

}
private void TaskMthodWithInParamOnly(bool val1, bool val2)
{
    Stopwatch timer = new Stopwatch();
    timer.Reset();
    timer.Start();
    while (timer.ElapsedMilliseconds < 20 * 1000)
    {
        Console.WriteLine("Elappsed time:" + timer.ElapsedMilliseconds.ToString());
    }
}
private bool TaskMthodWithReturnNoInParam()
{
      Stopwatch timer = new Stopwatch();
      timer.Reset();
      timer.Start();
      while (timer.ElapsedMilliseconds < 20 * 1000)
      {
          Console.WriteLine("Elappsed time:" + timer.ElapsedMilliseconds.ToString());
      }
      return true;
}
 
private bool TaskMthodWithReturnAndInParam(object val)
{
      Stopwatch timer = new Stopwatch();
      timer.Reset();
      timer.Start();
      while (timer.ElapsedMilliseconds < 20 * 1000)
      {
          Console.WriteLine(val.ToString());
      }
      return true;
}
}

如下内容引自https://www.cnblogs.com/liuqiyun/p/8110058.html

BeginInvoke 方法所调用的委托无论如何都是在 UI 线程中执行的。
那 BeginInvoke 究竟有什么用呢?
在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,具体的原因可以在看完我的这篇之后看看这篇:在多线程中如何调用Winform,如果你是大牛的话就不要看我这篇了,直接看那篇吧,反正那篇文章我没怎么看懂。
Invoke 和 BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。
正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。
而所谓的“一面响应操作,一面添加节点”永远只能是相对的,使 UI 线程的负担不至于太大而以,因为界面的正确更新始终要通过 UI 线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到 UI 线程中去做,这样也就达到了减轻 UI 线程负担的目的了。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/155090.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 二叉排序树(二)

    二叉排序树(二)

    2021年8月24日
    46
  • 最详细的Vuex教程

    最详细的Vuex教程最详细的Vuex教程什么是Vuex?vuex是一个专门为vue.js设计的集中式状态管理架构。状态?我把它理解为在data中的属性需要共享给其他vue组件使用的部分,就叫做状态。简单的说就是data中需要共用的属性。引入Vuex(前提是已经用Vue脚手架工具构建好项目)1、利用npm包管理工具,进行安装vuex。在控制命令行中输入下边的命令就可以了。npminstallvuex

    2022年6月9日
    29
  • 皮尔逊相关斯皮尔曼相关_肯德尔等级相关系数

    皮尔逊相关斯皮尔曼相关_肯德尔等级相关系数相关性、spss

    2022年8月4日
    11
  • tp-link路由器无线桥接详细教程_tp-link路由器怎么有线桥接

    tp-link路由器无线桥接详细教程_tp-link路由器怎么有线桥接本文介绍了TP-Link路由器有线桥接的设置方法,路由器有线桥接其实严格上应该叫做:两个(多个)路由器串联上网。主要适用于这样的网络环境:有A、B两台TP-Link路由器,A连接Moden(猫)上网,然后在用网线连接A和B,要实现B路由器也能够上网,包括B的无线网络。方法一、路由器B设置1、用网线连接电脑和TP-Link路由器B的A、B路由器之间,暂时不需要用网线连接。只让电脑连接无线路由器2、进…

    2025年7月15日
    6
  • java i586什么意思_linux里面i386 i686 i486 i586代表什么?是什么意思

    java i586什么意思_linux里面i386 i686 i486 i586代表什么?是什么意思在linux里面,我们经常会遇到i386i686i486I586这些代码,例如查看内核版本:[root@localhostlogs]#uname-aLinuxlocalhost.localdomain2.6.18-164.el5PAE#1SMPThuSep304:10:44EDT2009i686i686i386GNU/LinuxYouhavenewm…

    2022年5月30日
    41
  • C++ 编程之QueryInterface函数(一)

    C++ 编程之QueryInterface函数(一)前言组件对外公布的是接口;一个组件可以实现多个接口,也就是说可以对外公布多个接口,之前也总结过了,你很少会100%的去完全了解一个组件的所有接口,就像你去学习编程一样,你几乎不可能去成为编程中的全才。那么,既然我们不能去完全的了解一个组件提供的所有接口,那么我们在实际开发中,如何去判断一个组件是否提供对应的接口呢?看文档?是的,是个好主意,在文档的海洋,找到一个知识点,真的很难,浪费时间和

    2022年7月22日
    13

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号