大家好!我是付工。

大部分初学者在学习C#上位机编程时,多线程是一个很难逾越的鸿沟,不合理地使用多线程,会导致经常出现各种奇怪的问题,这也是很多初学者不敢使用多线程的原因。但是在实际开发中,多线程是一个不可避免的技术栈,基本上每个项目都会使用到,因此学好多线程技术,很重要。

一、多线程原理

首先,我们要了解什么是多线程,多线程是一种技术,能够让一个程序同时运行多个独立的执行流程,这个执行流程即线程,这样就可以提高程序的并发性和效率,使程序能够更有效地利用系统资源。

打个比方,刚创业的时候,我们可能是一个人身兼多职,既要对接业务,又要做技术,还需要管理财务,虽然你可以同时做这些事情,但毕竟只有一个人,这里的同时,其实是靠“时间管理”来实现的,这就是单核CPU实现多线程的原理,依靠时间片切换来实现多个任务,这个时间片很短,通常在10-100ms数量级,因此,让我们感觉是同时进行的。

随着技术的不断发展,现在的计算机CPU基本上都是多核的。8核、16核,都很常见,多核意味着有多个核心,可以同时运行多个任务。

因此,多核CPU上的多线程才是真正的多线程,它能让你的多段逻辑同时工作,多线程,可以真正发挥出多核CPU的优势来,达到充分利用CPU的目的。

如果你做项目不使用多线程,不仅是技术的问题,也是对电脑资源的极大浪费,就像买了一辆性能车,但是只用来日常代步。

二、多线程发展

我们知道了多线程技术可以保证我们代码的高效运行,提高CPU资源的使用率,为什么很多人不敢使用,主要是因为多线程如果使用不当,容易出现各种奇怪的问题。

我们必须要明确一点,多线程是“不可控的”,不要把多线程当做一个开关,需要的时候就开一下,不需要的时候就关闭一下。从微观角度来看,多线程是靠CPU调度来实现的,我们常说的开启多线程,只是告诉CPU,这个线程可以开了,但是至于是立即开,还是等一会再开,这个是由CPU调度决定的,对于关闭多线程也是一样。

我们后续提到的控制多线程启动、停止、暂停、继续,这些都是.NET框架中提供的一些接口(方法)给开发人员,这样程序员就可以间接地实现多线程。

微软的多线程技术也是在一直发展中,在.NET 1.0中就出现了多线程Thread,到2.0时推出了ThreadPool线程池,再到3.0是出现了Task,Task也是我们目前使用比较多的,Task被称之为多线程的最佳实践,再到4.0时推出Parallel并行编程,再到4.5推出async/await语法糖,它让我们可以用同步方法来实现异步编程。

三、多线程启停

Task是我们使用多线程开发中经常使用的一个类,这个类中提供了丰富的API函数,让我们可以很方便地对多线程进行管理,包括开启多线程了,就有很多种方法,比如Task.Run、Task.Factory.StartNew、Start等,由于篇幅有限,这里以其中一种进行说明。

我们来使用多线程实现一个简单的案例,我们来做一个线程任务,这个任务很简单,就是让一个值类型的变量,每间隔100ms,自增一次,到一个值后,再0开始重新计数,然后将这个值显示在界面上,界面如下所示:

所以该任务执行代码如下:

我们可以看到在方法里调用了一个cts对象,这个对象就是CancellationTokenSource的对象,因此我们需要创建一个CancellationTokenSource对象cts,同时在属性CurrentValue中,要显示控件的值,这里需要用到委托实现跨线程访问的问题,这个我们后续专题讲解,代码如下:

然后在启动线程按钮的事件里,编写代码如下:

停止线程按钮的事件里,只需要调用cts的Cancel方法即可:

我们可以看到,这里就是通过cts来控制cts的IsCancellationRequested属性,进而实现多线程的控制,这里的cts.IsCancellationRequested类似于一个布尔类型的标志位,但是CancellationTokenSource的作用不仅如此,还可以在此基础上实现多线程超时判断,注册事件等更复杂的多线程操作。

四、多线程暂停继续

多线程的暂停继续,.NET为我们提供了另外一个对象——ManualResetEvent,这个对象会有一个值,这个值是布尔类型,就像一个门闸一样,True是打开门闸,False是关闭门闸,所以想要暂停多线程就调用这个对象的Reset方法,想要继续多线程就调用这个对象的Set方法,使用非常简单。

首先我们创建一下这个对象,可以通过构造方法,给这个对象赋初始值,我这里为True,这样就能直接运行,不会阻塞,代码如下:

但是如果希望这个对象与多线程有所联系,必须要在多线程的方法里体现这个对象的作用,这个是调用这个对象的WaitOne方法,表示在调用的地方阻塞住,通过判断True或者False来决定是否继续执行,就像大家开车过高速收费站一样,即使现在普遍采用ETC了,在入口也需要减速,有一个ETC识别的过程,识别成功才会抬杆,识别不对,杆子是不会自动抬起的,这个是一样的道理。

所以线程执行代码修改如下:

对比一下,其实就是加了一个manual.WaitOne()。

线程暂停继续代码如下:

暂停继续的使用除了ManualResetEvent,还有一个AutoResetEvent,AutoResetEvent和ManualResetEvent的用法基本上是一样的,这里就不过多赘述,大家可以自己尝试一下。

这两者的区别在于一个是手动,一个是自动,AutoResetEvent会在置位之后自动复位,这样体现在多线程里,就是会只执行一次,就像大家进小区一样,如果有10辆车在排队,这时候如果自动模式,每次都要抬杆落杆,每次只允许进一辆车,如果是手动模式,可以由保安控制门闸打开,等10辆车都进去之后,再由保安将门闸关闭。

基于C#实现多线程启动停止暂停继续的更多相关文章

  1. Quartz的任务的临时启动和暂停和恢复

    Quartz的任务的临时启动和暂停和恢复 在项目中需要手动启停某些服务,那么需要有一个控制这些任务的类.由于任务是有Quartz控制的,我们只需要通过Quartz的相关的API实现相关的功能即可. p ...

  2. Linux Systemd——在RHEL/CentOS 7中启动/停止/重启服务

    RHEL/CentOS 7.0中一个最主要的改变,就是切换到了systemd.它用于替代红帽企业版Linux前任版本中的SysV和Upstart,对系统和服务进行管理.systemd兼容SysV和Li ...

  3. 在CentOS 7中启动/停止/重启服务

    RHEL/CentOS 7.0中一个最主要的改变,就是切换到了systemd.它用于替代红帽企业版Linux前任版本中的SysV和Upstart,对系统和服务进行管理.systemd兼容SysV和Li ...

  4. 控制Linux下 mono 服务的启动停止

    当Window下的服务部署到Linux的时候,我们一般用Mono.service 来启动停止.参数比较多,不太好用.于是有个这个Shell脚本. 用法:moa s1 start #启动         ...

  5. Linux Oracle服务启动&停止脚本与开机自启动

    在CentOS 6.3下安装完Oracle 10g R2,重开机之后,你会发现Oracle没有自行启动,这是正常的,因为在Linux下安装Oracle的确不会自行启动,必须要自行设定相关参数,首先先介 ...

  6. Python多线程启动http.server

    OS: Windows 8.1 with update 关键字:Python3.4, http.server, Thread 例子代码如下: import os from threading impo ...

  7. 批处理命令行CMD启动停止重启IIS的命令

    原文:批处理命令行CMD启动停止重启IIS的命令 启动IIS: net start iisadmin    (IIS的整个服务) net start w3svc       (WWW网页WEB服务) ...

  8. selenium之多线程启动grid分布式测试框架封装(四)

    九.工具类,启动所有远程服务的浏览器 在utils包中创建java类:LaunchAllRemoteBrowsers package com.lingfeng.utils; import java.n ...

  9. bat启动/停止oracle服务

    原文:bat启动/停止oracle服务 自己的电脑比较慢,尤其装了oracle10g后,服务开启和关闭用bat文件操作省事点 开启服务 @echo offnet start OracleService ...

  10. 绿色版Tomcat 启动 + 停止 + 随系统自动启动 - - 博客频道 - CSDN.NET

    body { font-family: "Microsoft YaHei UI","Microsoft YaHei",SimSun,"Segoe UI ...

随机推荐

  1. 【THUPC 2024 初赛】 E 转化

    [THUPC 2024 初赛] 转化 我都能做出来,那就是大水题了. 思路 首先我们要确定最大可以变色的球的数量 \(tot\). 有如下两个贪心步骤: 所有颜色使用分裂操作,并更新 \(a_i\). ...

  2. python 自动下载 moudle

    import sys,re,subprocess import os from subprocess import CalledProcessError new_set = set() ls = se ...

  3. Java垃圾回收器总结

    什么是Java垃圾回收器 Java垃圾回收器是Java虚拟机(JVM)的三个重要模块(另外两个是解释器和多线程机制)之一,为应用程序提供内存的自动分配(Memory Allocation).自动回收( ...

  4. 使用 BenchmarkDotNet 对 .NET 代码进行性能基准测试

    前言 在软件开发领域,性能基准测试是确保软件系统高效.稳定运行的重要环节.它可以帮助你评估应用程序的性能,了解其在不同条件下的响应时间.吞吐量.资源利用率等.通过基准测试,你可以确定系统在处理特定工作 ...

  5. 【java基础】-- java接口和抽象类的异同分析

    在java中,通常初学者搞不懂接口与抽象类,这也是面试比较容易问到的一个问题.下面我来谈谈自己的理解.如有不妥之处,还望批评指正,不胜感激. 目录 1.抽象类怎么定义和继承? 2.接口怎么定义和实现? ...

  6. javase学习文档

        javase学习文档(更新) javase 学习文档已更新 查看地址:https://lib.stazxr.cn/codenotes/java/javase/

  7. dotnet core微服务框架Jimu ~部署和调用演示

    首先运行 consul 下载 consul 以开发模式运行 consul agent -dev 2. 调试 用 Visual Studio 2022 IDE 打开项目: 右击解决方案-选择" ...

  8. Ant Design Pro项目一初始化就报a标签嵌套a标签错误<a> cannot as a descendant of <a>

    前情 公司经常需要做一些后台管理页面,我们选择了Ant Design Pro,它是基于 Ant Design 和 umi 的封装的一整套企业级中后台前端/设计解决方案. 坑位 按官方文挡一步步下来,项 ...

  9. Less使用备忘录

    定义 Less (Leaner Style Sheets 的缩写) 是一门向后兼容的 CSS 扩展语言,动态样式语言. 使用方式 直接引入less.js文件 好处:能获取客户端的数据,从而进行进一步的 ...

  10. 不求甚解--详解ansible-playbook中roles的用法(二)

    前言 本文将详细介绍ansible-playbook中roles的各种用法 环境准备 组件 版本 操作系统 Ubuntu 22.04.4 LTS ansible 2.17.6 基本用法 文件结构 . ...