Unity C#基础之 多线程的前世今生(上) 科普篇

多线程,项目中的应用不可缺少,能极大的提高程序的响应速度,但是也会提高内存和CPU的计算量(空间换时间),下面简单介绍下多线程从.NET 1.0版本到.NET 4.0版本的发展历程及使用示例

示例工程下载Unity 2017.3.0 P4 .NET版本4.6

在介绍之前先为大家科普下多线程的基础知识
  • 什么是进程
  • 什么是线程
  • 什么是多线程
  • 多线程的优点
  • 多线程的缺点
  • 何时使用多线程
  • 何时不要使用多线程
  • 同步和异步的区别

    什么是进程?

    进程是一个程序在电脑上运行时,全部计算资源的合集叫进程。当一个程序(例如QQ)开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源。而一个进程又是由多个线程所组成的。(QQ信息的接受和发送,图片接受发送,UI界面等)

    什么是线程?

    线程是程序的最小执行单位,包含计算资源,,任何一个操作的响应都是线程完成的。线程是程序中的一个执行流,每个线程都有自己的专有寄存器(栈指针、程序计数器等),但代码区是共享的,即不同的线程可以执行同样的函数。(例如QQ的UI界面就是一个线程处理,文字的收发一个线程处理,图片的收发一个线程处理)

    什么是多线程?

    多线程是指程序中包含多个执行流,即在一个程序中可以同时运行多个不同的线程来执行不同的任务,也就是说允许单个程序创建多个并行执行的线程来完成各自的任务。

    多线程的好处:

    可以提高CPU的利用率。在多线程程序中,一个线程必须等待的时候,CPU可以运行其它的线程而不是等待,这样就大大提高了程序的效率。多线程特点:不卡主线程、速度快、无序性

多线程的不利方面

  • 线程也是程序,所以线程需要占用内存,线程越多占用内存也越多;
  • 多线程需要协调和管理,所以需要CPU时间跟踪线程;
  • 线程之间对共享资源的访问会相互影响,必须解决竞用共享资源的问题(线程安全);
  • 线程太多会导致控制太复杂,最终可能造成很多Bug;

何时使用多线程

多线程程序一般被用来在后台执行耗时的任务。主线程保持运行,并且工作线程做它的后台工作。对于Windows Forms程序来说,如果主线程试图执行冗长的操作,键盘和鼠标的操作会变的迟钝,程序也会失去响应。由于这个原因,应该在工作线程中运行一个耗时任务时添加一个工作线程,即使在主线程上有一个有好的提示“处理中…”,以防止工作无法继续。这就避免了程序出现由操作系统提示的“没有相应”,来诱使用户强制结束程序的进程而导致错误。模式对话框还允许实现“取消”功能,允许继续接收事件,而实际的任务已被工作线程完成。BackgroundWorker恰好可以辅助完成这一功能。
在没有用户界面的程序里,比如说Windows Service, 多线程在当一个任务有潜在的耗时,因为它在等待另台电脑的响应(比如一个应用服务器,数据库服务器,或者一个客户端)的实现特别有意义。用工作线程完成任务意味着主线程可以立即做其它的事情。
另一个多线程的用途是在方法中完成一个复杂的计算工作。这个方法会在多核的电脑上运行的更快,如果工作量被多个线程分开的话(使用Environment.ProcessorCount属性来侦测处理芯片的数量)。
一个C#程序称为多线程的可以通过2种方式:明确地创建和运行多线程,或者使用.NET framework的暗中使用了多线程的特性——比如BackgroundWorker类, 线程池,threading timer,远程服务器,或Web Services或ASP.NET程序。在后面的情况,人们别无选择,必须使用多线程;一个单线程的ASP.NET web server不是太酷,即使有这样的事情;幸运的是,应用服务器中多线程是相当普遍的;唯一值得关心的是提供适当锁机制的静态变量问题。

何时不要使用多线程

多线程也同样会带来缺点,最大的问题是它使程序变的过于复杂,拥有多线程本身并不复杂,复杂是的线程的交互作用,这带来了无论是否交互是否是有意的,都会带来较长的开发周期,以及带来间歇性和非重复性的bugs。因此,要么多线程的交互设计简单一些,要么就根本不使用多线程。除非你有强烈的重写和调试欲望。
当用户频繁地分配和切换线程时,多线程会带来增加资源和CPU的开销。在某些情况下,太多的I/O操作是非常棘手的,当只有一个或两个工作线程要比有众多的线程在相同时间执行任务快的多。稍后我们将实现生产者/耗费者 队列,它提供了上述功能。

同步和异步的区别

同步方法调用在程序继续执行之前需要等待同步方法执行完毕返回结果(同一时间只能做一件事)
异步方法则在被调用之后立即返回以便程序在被调用方法完成其任务的同时执行其它操作(同一时间内可以并行做多件事)

话不多说,上Code,简单的异步多线程

    /// 
    /// 异步方法
    /// 
    private void AsyncOnClick()
    {
        Debug.Log($"AsyncOnClick - Start 线程ID:{Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");

        Action act = this.DoSomethingLong;        //act("同步菜鸟海澜");//写法等同于act.Invoke("菜鸟海澜");

        act.BeginInvoke("异步菜鸟海澜", null, null);

        Debug.Log($"AsyncOnClick - End 线程ID:{Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }
    /// 
    /// 同步方法
    /// 
    private void SyncOnClick()
    {
        Debug.Log($"SyncOnClick - Start 线程ID: {Thread.CurrentThread.ManagedThreadId.ToString("00")} 时间点: {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        this.DoSomethingLong("同步菜鸟海澜");
        Debug.Log($"SyncOnClick - End  线程ID: {Thread.CurrentThread.ManagedThreadId.ToString("00")} 时间点:  {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }


打印结果如下

调用顺序是先调用的异步然后是同步,但是打印结果是先出现的同步结果,说明异步没有卡主线程

异步多线程是无序的:启动无序 执行时间不确定 结束无序,但是要控制他的执行顺序一般分为三类 回调 等待 状态

回调

     /// 
    /// 异步回调  
    /// 
    private void AsyncCallback()
    {
        #region 无返回值
        {
            Action act = this.DoSomethingLong;
            IAsyncResult iAsyncResult = null;
            AsyncCallback callback = ar =>
            {
                Debug.Log(object.ReferenceEquals(ar, iAsyncResult));
                Debug.Log(ar.AsyncState);
                Debug.Log($"这里是BeginInvoke的回调{Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            };
            iAsyncResult = act.BeginInvoke("AsyncOnClick01", callback, "菜鸟海澜");
        }
        #endregion
        #region 有返回值
        {
            Func func = i => i.ToString();
            //AsyncCallback
            IAsyncResult iAsyncResult = func.BeginInvoke(DateTime.Now.Year, ar =>
            {
                string resultIn = func.EndInvoke(ar);//对于每个异步操作,只能调用一次 EndInvoke。调用EndInvoke后立即强制释放线程,否则系统在合适时机释放线程
                Debug.Log($"This is {ar.AsyncState} 的异步调用结果 {resultIn}");
            }, "回调传参菜鸟海澜");
            //string result = func.EndInvoke(iAsyncResult);//获取返回值 不在回调中调用EndInvoke卡主线程
        }
        #endregion
    }

打印结果如下

等待

    /// 
    /// 异步等待
    /// 
    private void AsyncWaitOnClick()
    {
        {
            Action act = this.DoSomethingLong;
            IAsyncResult iAsyncResult = act.BeginInvoke("btnAsync_Click", null, null);

            int i = 1;
            while (!iAsyncResult.IsCompleted)//1 卡界面,主线程在等待   2 边等待边做事儿  3有误差
            {
                if (i < 10)
                {
                    Debug.Log($"文件上传{i++ * 10}%。。。请等待");
                }
                else
                {
                    Debug.Log("已完成99%。。。马上结束");
                }
                Thread.Sleep(200);
            }
            Debug.Log("文件上传成功!!!");

            act.EndInvoke(iAsyncResult);//等待
            Debug.Log("这里是BeginInvoke调用完成之后才执行的。。。");
        }
    }

打印结果如下

状态

    /// 
    /// 异步状态
    /// 
    private void AsyncStateOnClick()
    {
        Debug.Log($"AsyncStateOnClick Start {name}  线程ID:{Thread.CurrentThread.ManagedThreadId.ToString("00")} 时间点: {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        Action act = this.DoSomethingLong;
        IAsyncResult iAsyncResult = act.BeginInvoke("btnAsync_Click", null, null);
        //iAsyncResult.AsyncWaitHandle.WaitOne();//一直等待任务完成,第一时间进入下一行
        //iAsyncResult.AsyncWaitHandle.WaitOne(-1);//一直等待任务完成,第一时间进入下一行
        iAsyncResult.AsyncWaitHandle.WaitOne(1000);//最多等待1000ms,否则就进入下一行,可以做一些超时控制
        Debug.Log($"AsyncStateOnClick End {name}  线程ID:{Thread.CurrentThread.ManagedThreadId.ToString("00")} 时间点: {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }

打印结果如下

发表评论