多线程系列(一)多线程基础
线程相关概念
在学习多线程之前,先来了解下几个与多线程相关的概念。
进程:进程是计算机的概念,程序在服务器运行时占据全部计算资源的总和,一个应用程序运行起来就是一个进程,打开windows的任务管理器,如下图
线程:线程也是计算机的概念,线程是进程的最小单位,也是程序在响应操作系统时的最小单位,一个进程至少由一个线程(主线程)构成。线程和进程一样也会占据一定的CPU、内存、网络、硬盘IO等。一个线程隶属于某个进程,进程销毁,线程也随之销毁。
句柄:是一个long类型的数字,是操作系统用来标识应用程序的,有点主键或者身份证号码的意思。
多线程:一个进程或者说一个应用程序有多个线程在运行参与计算。
C#里面的多线程
Thread类是C#语言对线程对象的封装。在.netframework1.0开始出现。在后面的多线程系列文章中会讲到在不同的.netframework版本中多线程的API使用,在本篇文章中,先来初步认识多线程。
为什么可以使用多线程
1:CPU的多核技术和模拟核技术:
如计算机的参数概念4核8线程,所谓的4核8线程,4核指的是物理核心。通过超线程技术,用一个物理核模拟两个虚拟核,每个核两个线程,总数为8线程。四核八线程采用的超线程技术,是指每个CPU核心没有满负荷运载时,其剩余用量可以模拟成虚拟的核心。单个物理核同一时间点只能处理一个线程,通过超线程技术可以实现单个物理核实现线程级别的并行计算。
2:CPU分片:实际上CPU在同一时刻只能处理一个任务,但是因为CPU的计算能力强大,在1秒内可以响应不同的任务,把1秒的处理能力分成10份,1到100毫秒处理任务A,101到200毫秒处理任务B,201到300毫秒处理任务C…,从宏观角度来看,感觉多个任务在并发执行,这个就是CPU的分片。
初识同步和异步
同步方法:发起调用,完成后才继续下一行;非常符合开发思维,有序由上至下执行;
异步方法:发起调用,不用等待完成,直接进入下一行,启用一个新的线程来完成计算。
同步方法就像真诚的请人吃饭,客人说他有事,需要忙一会儿,请吃饭的人等待客人忙完了再一起吃饭。异步方法就像客气的请人吃饭,客人说他有事,请吃饭的人说那你先去忙吧,然后自己就去吃饭了。
同步和异步的比较
同步方法卡界面,主线程(UI)线程忙于计算,无暇他顾,异步方法不卡界面:主线程闲置,计算任务交给子线程完成,改善用户体验。如在winform中点击按钮采用同步的方式调用一个复杂的任务计算会导致界面短暂卡死,直到任务计算结束才可以操作界面。
在web应用中发个短信通知,记录一个日志,都可以采用异步的方式去执行,客户端不用等到短信发送成功或者日志记录成功才能接受到服务端的响应。
为了能够清楚的说明情况,这里采用测试程序对比的方式,测试程序界面如下:
计算任务:
private void DoSomeThing(string btnName) {
Console.WriteLine($"{btnName} 开始,当前线程id:{Thread.CurrentThread.ManagedThreadId}");
long lResult = 0;
for (long i = 0; i < 1_000_000_000; i++)
{
lResult += i;
}
Console.WriteLine($"{btnName} 结束,当前线程id:{Thread.CurrentThread.ManagedThreadId}");
}
同步方式调用:
private void BtnSync_Click(object sender, EventArgs e)
{
Console.WriteLine($"btnSync_Click Start {Thread.CurrentThread.ManagedThreadId.ToString("00")}" +
$" {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
Stopwatch stopwatch = new Stopwatch();
stopwatch.Start();
for (int i = 0; i < 5; i++)
{
string name = string.Format($"btnSync_Click_{i}");
this.DoSomeThing(name);
}
stopwatch.Stop();
Console.WriteLine($"同步方法耗时:{stopwatch.ElapsedMilliseconds}");
Console.WriteLine($"DoSomeThing End {Thread.CurrentThread.ManagedThreadId.ToString("00")} " +
$"{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
}
同步方式调用执行结果:
同步方式调用时CPU的使用情况:
异步方式调用:
private void BtnAsync_Click(object sender, EventArgs e)
{
Console.WriteLine($"btnAsync_Click Start {Thread.CurrentThread.ManagedThreadId.ToString("00")}" +
$" {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
Stopwatch stopwatch = new Stopwatch();
List<Task> tasks = new List<Task>();
stopwatch.Start();
for (int i = 0; i < 5; i++)
{
int k = i;
tasks.Add(Task.Run(()=> {
string name = string.Format($"btnAsync_Click{k}");
this.DoSomeThing(name);
}));
}
Task.Run(()=> {
Task.WaitAll(tasks.ToArray());
stopwatch.Stop();
Console.WriteLine($"异步方法耗时:{stopwatch.ElapsedMilliseconds}");
Console.WriteLine($"DoSomeThing End {Thread.CurrentThread.ManagedThreadId.ToString("00")}" +
$" {DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
});
}
异步方式调用执行结果:
异步方式调用时CPU的使用情况:
总结
同步方法慢,上图耗时(16402毫秒),因为只有一个线程计算。异步方法快,上图耗时(10524毫秒),因为有多个线程参与计算。观察同步和异步调用时的使用情况折线图分析得知:多线程其实就是资源换取性能。在一个应用程序中是不是开启的线程越多越好?不是的,因为开启更多的资源,需要消耗更多的计算机资源,资源是有限的、资源调度也需要消耗资源,就像项目经理管理开发人员保证项目进度,项目经理调度(管理)开发人员也是需要工资的。
一个订单表统计很耗时间,能用多线程优化性能么?不能!这操作只包含一个任务,没办法并行计算,就像一个老师不能同时在两个班级讲课。如果一个操作在查询数据库的同时,需要调用接口、读写硬盘文件、做数据计算,这个可以用多线程优化性能,因为多个任务可以并行计算。
同步方法有序进行,异步多线程无序
启动无序:线程资源是属于非托管资源,是程序向操作系统申请的,由操作系统的调度策略决定,所以启动顺序是随机的,cpu使用同一个线程计算同一个任务,执行时间也是不确定的,so,结束也是无序的。在使用多线程的时候一定要小心,尤其是多线程间有顺序要求的时候通过延迟一点时间(Thread.Sleep())来控制执行顺序,这是不靠谱的。
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(12)-系统日志和异常的处理②
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(14)-EasyUI缺陷修复与扩展
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(10)-系统菜单栏[附源码]
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(8)-MVC与EasyUI DataGrid 分页
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(5)-EF增删改查
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(15)-权限管理系统准备
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(11)-系统日志和异常的处理①
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(9)-MVC与EasyUI结合增删改查
- No.3 啥是数据运营(三):思维方式
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(7)-MVC与EasyUI DataGrid
- MySQL数据同步【双主热备】
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(6)-Unity 依赖注入
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(4)-创建项目解决方案
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(3)-漂亮系统登陆界面
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 五分钟快速搭建Serverless免费邮件服务
- 基于qiankun落地部署微前端爬”坑“记
- Android推送的群魔乱舞
- 用百度接口实现图片文字识别,并打包成安装包软件
- 视野前端(二)V8引擎是如何工作的
- 【干货】Chrome插件(扩展)开发全攻略
- 超性感的React Hooks(一):为何她独具魅力
- 超性感的React Hooks(二)再谈闭包
- Python全栈(一)基础之11.函数(3)
- Python全栈(二)数据结构和算法之1.算法和数据结构引入
- Android开发(第一行代码 第二版) 常见异常和解决办法(基于Android Studio)(一)
- Python SQLite 基本操作和经验技巧(一)
- Python字典及基本操作(超级详细)
- Python matplotlab库使用方法及注意事项
- 超性感的React Hooks(三):useState