C# .Net学习笔记—— 异步和多线程(Async和Sync)

一、概念

进程:一个程序运行时,占用的全部计算资源的总和

线程:1、程序执行流的最小单位;任何操作都是由线程完成的

2、线程是依托于进程存在的,一个进程可以包含多个线程;

3、线程也可以有自己的计算资源

多线程:多个执行流同时运行

1、CPU太快了,分时间片-一上下文切换(加载环境--计算-一保存环境)

微观角度,一个核同一时刻只能执行一个线程;宏观的来说是多线程并发

2、多CPU多核可以独立工作

4核8线程一核是物理的核线程是指虚拟核

同步:完成计算之后,再进入下一行

异步:不会等待方法的完成,会直接进入下一行非阻塞

C#异步和多线程有什么差别

多线程就是多个thread并发;

异步是硬件式的异步

异步多线程-thread pool task

二、同步异步简单案例

cs 复制代码
 private void buttonSync_Click(object sender, EventArgs e)
        {
            Log.Info($"S Start {Thread.CurrentThread.ManagedThreadId}");
            int j = 3;
            int k = 5;
            int m = j + k;
            for (int i = 0; i < 5; i++)
            {
                string name = string.Format($"**********S{i}*************");
                this.DoSomethingLong(name);
            }
            Log.Info($"S End");
        }

        private void buttonAsync_Click(object sender, EventArgs e)
        {
            Log.Info($"\nA Start {Thread.CurrentThread.ManagedThreadId}");
            Action<string> action = this.DoSomethingLong;
            for (int i = 0; i < 5; i++)
            {
                string name = string.Format($"**********A{i}*************");
                action.BeginInvoke(name, null, null);
            }
            Log.Info($"A End");
        }

        private void DoSomethingLong(string name)
        {
            Log.Info($"DoSomethingLong Start {Thread.CurrentThread.ManagedThreadId.ToString("00")} {name} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
            long result = 0;
            for (int i = 0; i < 100000000; i++)
            {
                result += i;
            }
            Thread.Sleep(2000);
            Log.Info($"DoSomethingLong End {Thread.CurrentThread.ManagedThreadId.ToString("00")} {name} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        }

同步:

异步:

从打印结果来看也能反映出,

同步:1、完成计算之后,再进入下一行。

2、同步方法卡界面,主(UI)线程忙于计算。

3、同步方法慢,只有一个线程干活
异步:1、不会等待方法的完成,会直接进入下一行非阻塞。

2、异步多线程方法不卡界面,主线程完事,计算任务交给子线程在做。

3、异步多线程方法快,因为多个线程并发运算

4、异步并不是线性增长,而是资源换时间,多线程也有管理成本

5、提升用户体验

6、并不是越多越好

7、最好是多个独立任务同时进行的使用

异步等待方式1:异步回调:

cs 复制代码
        private void buttonAysncCallback_Click(object sender, EventArgs e)
        {
            Log.Info($"DoSomethingLong Start {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
            Action<string> action = this.DoSomethingLong;

            AsyncCallback callback = ia => Log.Info($"计算完成。{Thread.CurrentThread.ManagedThreadId.ToString("00")}  {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
            action.BeginInvoke("AysncCallBack", callback, null);
            Log.Info($"DoSomethingLong End {Thread.CurrentThread.ManagedThreadId.ToString("00")} {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
        }

从结果来看,异步回调action.BeginInvoke()可以得到一个AsyncCallBack,然后这个callBack再执行的委托。这样可以有效的控制我们线程的次序。

缺点:我想要全部计算都完成了再给用户返回结果,这种方法实现不了

异步等待方式2:线程等待

cs 复制代码
asyncResult.AsyncWaitHandle.WaitOne(); //等待任务完成
asyncResult.AsyncWaitHandle.WaitOne(); //等待任务完成
asyncResult.AsyncWaitHandle.WaitOne(1000); //等待;但是最多等待1000ns

异步等待方式3: EndInvoke()

cs 复制代码
        private void buttonTest_Click(object sender, EventArgs e)
        {
            Func<int> func = () =>
            {
                Thread.Sleep(2000);
                return DateTime.Now.Day;
            };
            Console.WriteLine($"func.Invoke()={func.Invoke()}");

            IAsyncResult asyncResult = func.BeginInvoke(r =>
            {
                Console.WriteLine(r.AsyncState);
            }, "哈啊哈哈");
            Console.WriteLine($"func.EndInvoke(asyncResult) = {func.EndInvoke(asyncResult)}");
        }
相关推荐
陈平安Java and C41 分钟前
MyBatisPlus
java
秋野酱1 小时前
如何在 Spring Boot 中实现自定义属性
java·数据库·spring boot
Bunny02122 小时前
SpringMVC笔记
java·redis·笔记
架构文摘JGWZ2 小时前
FastJson很快,有什么用?
后端·学习
feng_blog66882 小时前
【docker-1】快速入门docker
java·docker·eureka
枫叶落雨2224 小时前
04JavaWeb——Maven-SpringBootWeb入门
java·maven
m0_748232394 小时前
SpringMVC新版本踩坑[已解决]
java
码农小灰4 小时前
Spring MVC中HandlerInterceptor和Filter的区别
java·spring·mvc
量子-Alex4 小时前
【多视图学习】显式视图-标签问题:多视图聚类的多方面互补性研究
学习
乔木剑衣5 小时前
Java集合学习:HashMap的原理
java·学习·哈希算法·集合