Unity C#基础之 多线程的前世今生(下) 扩展篇

在前面两篇Unity C#基础之 多线程的前世今生(上) 科普篇Unity C#基础之 多线程的前世今生(中) 进阶篇中,相信大家对多线程有了一定的了解,这篇再详细的聊一聊在使用多线程中需要注意的地方~

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

本篇知识点

  • 异常处理
  • 线程取消 CancellationTokenSource
  • 多线程临时变量
  • 线程安全 lock
  • 语法糖 await async
异常处理

首先我们先执行下面一段代码 循环20次用Task线程执行以下Code,当执行循环 i=11i=12时抛出异常

img_bff22709381b53e2837a681ac1b99ddb.png

打印信息中没有 11、12的打印信息,也没有抛出异常的信息,这是因为主线程的Trycatch已经跳过
img_bd5b10e59a2513cb2c5b5b4970559b1a.png

然后我们在try块中添加 Task.WaitAll(taskList.ToArray());

img_e2bf3463f406c1df001f186d7cc2c4d7.png
打印信息如下 出现抛出异常信息
img_e59ba106e1a9bc9c34fd3afd1a230de5.png

img_2b64273f54b5fbc04ee1b080bd095a6b.png
然后我们去掉try块中的 Task.WaitAll(taskList.ToArray()); 在每个线程中添加Try Catch
img_c7e39b141af0b4a9e4e033e438deb217.png

打印结果可以捕捉到异常 所以要想捕捉到异常且不卡界面,捕捉异常最好在每个自己的线程中捕捉

img_95eabd2287c6a4b91efd8e15b3b6c3c4.png

相应的全部Code
    private void TryCatchOnClick()
    {
        System.Diagnostics.Stopwatch watch = new System.Diagnostics.Stopwatch();
        watch.Start();
        Debug.Log($"TryCatchOnClick Start {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        try
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();

            #region 异常处理
            //多线程里的异常是会被吞掉,除非waitall
            // 建议 多线程里面,是不允许异常的,也就是内部try catch,自己处理好
            for (int i = 0; i < 20; i++)
            {
                string name = string.Format($"TryCatchOnClick{i}");
                Action<object> act = t =>
                {
                    try
                    {
                        Thread.Sleep(2000);
                        if (t.ToString().Equals("TryCatchOnClick11"))
                        {
                            throw new Exception(string.Format($"{t} 执行失败"));
                        }
                        if (t.ToString().Equals("TryCatchOnClick12"))
                        {
                            throw new Exception(string.Format($"{t} 执行失败"));
                        }
                        Debug.Log($"{t} 执行成功");
                    }
                    catch (Exception ex)
                    {
                        Debug.Log(ex.Message);
                    }
                };
                taskList.Add(taskFactory.StartNew(act, name));
            }
            //Task.WaitAll(taskList.ToArray());
            #endregion
        }
        catch (AggregateException aex)
        {
            foreach (var item in aex.InnerExceptions)
            {
                Debug.Log(item.Message);
            }
        }
        catch (Exception ex)
        {
            Debug.Log(ex.Message);
        }

        watch.Stop();
        Debug.Log($"TryCatchOnClick   End {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }

线程取消

某个线程达到预期效果后需要取消其他的线程,我们可以用 CancellationTokenSource,当然可以用一个共享的bool变量,但是CancellationTokenSource的好处是可以让没来的及启动的线程直接取消,从根本上取消启动

    private void TaskCancel()
    {
        System.Diagnostics.Stopwatch watch = new System.Diagnostics.Stopwatch();
        watch.Start();
        Debug.Log($"TaskCancel Start {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}TaskCancel");
        try
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();

            //线程取消不是操作线程,而是操作信号量(共享变量,多个线程都能访问到的东西,变量/数据库的数据/硬盘数据)
            //每个线程在执行的过程中,经常去查看下这个信号量,然后自己结束自己
            //线程不能别人终止,只能自己干掉自己,延迟是少不了的
            //CancellationTokenSource可以在cancel后,取消没有启动的任务
            CancellationTokenSource cts = new CancellationTokenSource();//bool值
            for (int i = 0; i < 200; i++)
            {
                string name = string.Format("btnThreadCore_Click{0}", i);
                Action<object> act = t =>
                {
                    try
                    {
                        Thread.Sleep(2000);
                        if (t.ToString().Equals("btnThreadCore_Click11"))
                        {
                            throw new Exception(string.Format("{0} 执行失败", t));
                        }
                        if (t.ToString().Equals("btnThreadCore_Click12"))
                        {
                            throw new Exception(string.Format("{0} 执行失败", t));
                        }
                        if (cts.IsCancellationRequested)//检查信号量
                        {
                            Debug.Log($"{t} 放弃执行");
                            return;
                        }
                        else
                        {
                            Debug.Log($"{t} 执行成功");
                        }
                    }
                    catch (Exception ex)
                    {
                        cts.Cancel();
                        Debug.Log(ex.Message);
                    }
                };
                taskList.Add(taskFactory.StartNew(act, name, cts.Token));
            }
            Task.WaitAll(taskList.ToArray());
        }
        catch (AggregateException aex)
        {
            foreach (var item in aex.InnerExceptions)
            {
                Debug.Log(item.Message);
            }
        }
        catch (Exception ex)
        {
            Debug.Log(ex.Message);
        }

        watch.Stop();
        Debug.Log($"TaskCancel   End {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }
打印结果如下
img_c91cbaa8ebe918660895435ab41d6b61.png

img_d48bf0c44100318a43a792a4250ff829.png

多线程临时变量

这个就比较简单了,直接上Code

    private void TaskTempVariable()
    {
        System.Diagnostics.Stopwatch watch = new System.Diagnostics.Stopwatch();
        watch.Start();
        Debug.Log($"TaskTempVariable Start {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        try
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();
            i  只有一个,真实实行的时候,已经是5了,
            k  多个k,每次是独立的k,跟i没关系
            int k;
            for (int i = 0; i < 5; i++)
            {
                int k = i;
                new Action(() =>
                {
                    Thread.Sleep(1000);
                    Debug.Log($"对应的数值K:{k}");
                    Debug.Log($"对应的数值I:{i}");
                }).BeginInvoke(null, null);
            }
            Task.WaitAll(taskList.ToArray());
        }
        catch (AggregateException aex)
        {
            foreach (var item in aex.InnerExceptions)
            {
                Debug.Log(item.Message);
            }
        }
        catch (Exception ex)
        {
            Debug.Log(ex.Message);
        }

        watch.Stop();
        Debug.Log($"TaskTempVariable   End {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }
打印结果
img_6fafc4a284596921a57e5a2c188ee25e.png

线程安全 lock

ConcurrentDictionary多线程版字典

多线程1000次操作一个int和List

    private static readonly object StaticAsyncLock = new object();
    private int TotalCount = 0;//
    private List<int> IntList = new List<int>(20000);
    private void TaskSafe()
    {
        System.Diagnostics.Stopwatch watch = new System.Diagnostics.Stopwatch();
        watch.Start();
        Debug.Log($"TaskSafe Start {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        try
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();
            //共有变量:都能访问局部变量/全局变量/数据库的一个值/硬盘文件
            //线程内部不共享的是安全

            //解决多线程冲突第一个办法:lock   ,lock的方法块儿里面是单线程的,所以将整个方法Lock多线程将变得毫无意义;lock里面的代码要尽量的少
            //解决多线程冲突第二个办法:没有冲突,从数据上隔离开
            for (int i = 0; i < 10000; i++)
            {
                int TempI = i;
                taskList.Add(taskFactory.StartNew(() =>
                {
                    lock (StaticAsyncLock)//lock后的方法块,任意时刻只有一个线程可以进入语法糖  lock(StaticAsyncLock) 编译后等于 Monitor.Enter(StaticAsyncLock)
                    {   //这里就是单线程
                        this.TotalCount += 1;//多个线程同时操作,有时候操作被覆盖了
                        IntList.Add(TempI);
                    }
                }));
                //语法糖 lock(StaticAsyncLock) 编译后等于 Monitor.Enter(StaticAsyncLock) Monitor.Exit(StaticAsyncLock) 
                //检查下这个变量(引用) 有没有被lock   有就等着,没有就占用,然后进去执行,执行完了释放
                //lock(this) 锁定当前实例,别的地方如果要使用这个实力里面的其他变量,则都被锁定了无法使用(不推荐这么写) 
                //如果每个实例想要单独的锁定  private object
                //string a="123456" lock(a)  string b="123456" 享元模式的内存分配,字符串值是唯一的,会锁定别的变量b
                //private static readonly object StaticAsyncLock = new object();
            }
            Task.WaitAll(taskList.ToArray());

            Debug.Log(this.TotalCount);
            Debug.Log(IntList.Count);
        }
        catch (AggregateException aex)
        {
            foreach (var item in aex.InnerExceptions)
            {
                Debug.Log(item.Message);
            }
        }
        catch (Exception ex)
        {
            Debug.Log(ex.Message);
        }

        watch.Stop();
        Debug.Log($"TaskSafe  End {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
    }
添加Lock锁前后打印结果
img_91599f9bef1578153ea90075c5855d1e.png

img_4918194c02e04fec0fee785dedde272a.png

语法糖 await async (C#5.0 .NET 4.5 CLR 4.0)

如果用一句话简单概括await async,那就是:多线程版的协程

先来一个简单的示例,做一个大象装冰箱,最后咱们再详细的分析
img_85e12026dbd6337f3e81d9d6c38ffa24.png

img_4be15c23e036c19a227b89ef3c42cfa3.png
img_4c385651660949dd68779d71aa9a2b23.png
打印结果
img_fa00c5b99b66ed93d353571ecffe1ae6.png
是不是很有趣?这种书写逻辑基本上和原来的协程一样,而且是真正的多线程,但是依据不能运行UnityEngine中的组件(例如:GameObject),下面我们详细的说一说 await async

第一个示例 在基础的方法上添加关键字 async 根据提示只有aysnc没有await 会有一个警告,跟普通方法没有区别(不得不说VS2017还是很不错的,提示、自动修补都很友好)

img_641d02fe314fb5dd697ca4d14b2a4fa9.png

打印结果
img_f4ad85653406f78fd8222b912b3df70d.png

第二个是示例 在第一个示例的基础上添加await关键字,这时当主线程到达await task时就返回了,继续执行方法外部的函数,可以理解为unity协程中的yeild reture,当task的线程块执行完毕后再回调 awati task后面的函数部分,这个回调的线程是不确定的:可能是主线程 可能是子线程 也可能是其他线程

    /// <summary>
    /// async/await 
    /// 不能单独await
    /// await 只能放在task前面
    /// 不推荐void返回值,使用Task来代替
    /// Task和Task<T>能够使用await, Task.WhenAny, Task.WhenAll等方式组合使用。Async Void 不行
    /// </summary>
    private  async void NoReturn()
    {
        //主线程执行
       Debug.Log($"NoReturn Sleep before await,线程ID:{Thread.CurrentThread.ManagedThreadId}");
        TaskFactory taskFactory = new TaskFactory();
        Task task = taskFactory.StartNew(() =>
        {
           Debug.Log($"多线程 Sleep before,线程ID:{Thread.CurrentThread.ManagedThreadId}");
            Thread.Sleep(3000);
           Debug.Log($"多线程 Sleep after,线程ID:{Thread.CurrentThread.ManagedThreadId}");
        });
        await task;//主线程到这里就返回了,执行主线程任务


        //子线程执行   其实是封装成委托,在task之后成为回调(编译器功能  状态机实现)
        //task.ContinueWith()
        //这个回调的线程是不确定的:可能是主线程  可能是子线程  也可能是其他线程
       Debug.Log($"NoReturn Sleep after await,线程ID:{Thread.CurrentThread.ManagedThreadId}");
    }

打印结果
img_c5d36978e19351cb5359c349c91f9c0e.png

第三个示例,如果需要获取这个返回的线程怎么办呢?直接把void 换成Task

    /// <summary>
    /// 无返回值  async Task == async void
    /// Task和Task<T>能够使用await, Task.WhenAny, Task.WhenAll等方式组合使用。Async Void 不行
    /// </summary>
    /// <returns></returns>
    private  async Task NoReturnTask()
    {
        //这里还是主线程的id
       Debug.Log($"NoReturnTask Sleep before await,线程ID:{Thread.CurrentThread.ManagedThreadId}");

        await  Task.Run(() =>
        {
           Debug.Log($"多线程 Sleep before,线程ID:{Thread.CurrentThread.ManagedThreadId}");
            Thread.Sleep(9000);
           Debug.Log($"多线程 Sleep after,线程ID:{Thread.CurrentThread.ManagedThreadId}");
        });

       Debug.Log($"NoReturnTask Sleep after await,线程ID:{Thread.CurrentThread.ManagedThreadId}");

        //return new TaskFactory().StartNew(() => { });  //不能return  没有async才行
    }
然后执行
img_7fca65416a936099a820379f5f24cff8.png
打印结果
img_4b297f334a6dc22cbc2db72861897ada.png

第四个示例,也是最终版的示例 返回线程+返回数值 ,获取一个Task<T> ,其中T就是返回值的类型

    /// <summary>
    /// 带返回值的Task  
    /// 要使用返回值就一定要等子线程计算完毕 卡线程
    /// </summary>
    /// <returns>async 就只返回long</returns>
    private async Task<long> FinallyAsync()
    {
        Debug.Log($"SumAsync  start 线程ID:{Thread.CurrentThread.ManagedThreadId}");
        long result = 0;

        await Task.Run(() =>
        {

            Debug.Log($"SumAsync await Task.Run 线程ID:{Thread.CurrentThread.ManagedThreadId}");
            Thread.Sleep(1000);

            for (long i = 0; i < 999999999; i++)
            {
                result += i;
            }
        });
        return result;
    }
然后执行
img_760d8d820b27a6c96cf2a42ca4e5699a.png
打印结果
img_20cb1867652472d9e3472845a9ad642f.png
非await版返回值 这种主线程获取result不会卡死
    /// <summary>
    /// 真的返回Task  不是async  
    /// 
    /// 要使用返回值就一定要等子线程计算完毕
    /// </summary>
    /// <returns>没有async Task</returns>
    private Task<int> TaskReturn()
    {
        Debug.Log($"TaskReturn  start 线程ID:{Thread.CurrentThread.ManagedThreadId}");
        TaskFactory taskFactory = new TaskFactory();
        Task<int> iResult = taskFactory.StartNew<int>(() =>
        {
            Thread.Sleep(3000);
            Debug.Log($"TaskReturn  Task.Run 线程ID:{Thread.CurrentThread.ManagedThreadId}");
            return 123;
        });

        Debug.Log($"TaskReturn    end 线程ID:{Thread.CurrentThread.ManagedThreadId}");
        return iResult;
    }
执行函数
img_ada8e59ca8412e38e4f850cb025d8253.png
打印结果
img_0cd2848385ea4e8f6da5bd394c1d18a0.png

以上就是await async 的相关内容,有需要补充的东西欢迎留言

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值