C#基础-线程暂停方案之重置事件

默认情况下,一个线程的特定指令相对于另一个线程中的指令的执行时机是不确定的,如果想要对这种不确定性进行控制,其中一种办法就是使用重置事件(虽然称为事件,但是跟C#的委托跟事件没关系)。

重置事件用于强迫代码等候另一个线程的执行,直到获得事件已发生的通知。

重置事件类型包括ManualResetEventManualResetEventSlim以及AutoResetEvent,但在使用过程中应该尽量用前面两种类型,而避免使用AutoResetEvent

ManualResetEvent

ManualResetEvent对象具有两种状态,非信号状态信号状态

  • ManualResetEvent对象处于非信号状态,调用WaitOne方法,则当前线程会被阻塞,直到此ManualResetEvent对象调用Set方法后线程才被释放。
  • ManualResetEvent对象处于信号状态,调用WaitOne方法时,当前线程不会被阻塞,ManualResetEvent立即释放线程并返回到非信号状态。

一、常用成员

构造函数

ManualResetEvent(Boolean)ManualResetEvent的构造函数,参数为false则创建的ManualResetEvent对象为非信号状态,反之为信号状态。

常用方法

boolean WaitOne():如果当前ManualResetEvent对象为非信号状态,阻塞当前线程,直到收到信号(调用Set()方法);如果当前的ManualResetEvent对象为信号状态,则不会阻塞当前线程。

  • 收到信号或当前对象为信号状态时返回true,否则一直等待,不会返回。

bool WaitOne([int millisecondsTimeout]):如果当前ManualResetEvent对象为非信号状态,阻止当前线程,直到超过指定时间或收到信号(某处调用了此对象的Set());如果当前的ManualResetEvent对象为信号状态,则不会阻塞当前线程。

  • 收到信号返回true,否则返回false

bool Set():将状态设置为信号状态,允许一个或多个等待线程继续。

  • 操作成功返回true,否则false

bool Reset():将状态设置为非信号状态。

  • 如果该操作成功,返回true,否则false

Dispose():释放WaitHandle类(ManualResetEvent为其子类)的当前实例所使用的所有资源。

二、示例

ManualResetEvent manualResetEvent= new ManualResetEvent(false);Socket server = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
server.Bind(new IPEndPoint(IPAddress.Any, 9090));
server.Listen();server.BeginAccept(new AsyncCallback(asyncResult =>
{Socket s = asyncResult.AsyncState as Socket;Socket ct = s.EndAccept(asyncResult);while (true){manualResetEvent.Reset();byte[] buffer = new byte[1024];ct.BeginReceive(buffer, 0, 1024, SocketFlags.None, new AsyncCallback(asyncResult =>{Socket c = asyncResult.AsyncState as Socket;c.EndReceive(asyncResult);manualResetEvent.Set();}), ct);manualResetEvent.WaitOne();Console.WriteLine(Encoding.UTF8.GetString(buffer));}
}), server);Socket client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
client.Connect(new IPEndPoint(IPAddress.Parse("127.0.0.1"), 9090));
for (int i = 0; i < 10; i++)
{//Thread.Sleep(20); //如果有个延时就可以看到不粘包client.Send(Encoding.UTF8.GetBytes("Hello"));
}Console.ReadLine();

三、注意

释放资源

官方文档中说明:实现 IDisposable接口。 在使用完类型后,应直接或间接释放类型。 若要直接释放类型,请在 try/catch块中调用其Dispose方法。 若要间接释放类型,请使用 using。 因此在使用过程中,可以的话尽量考虑资源的释放。

ManualResetEventSlim

ManualResetEventManualResetEventSlim的区别在于,前者默认使用核心同步,而后者进行了优化,除非万不得已,否则会尽量避免使用核心机制,因此ManualResetEventSlim的性能更好。因此一般情况下应选用ManualResetEventSlim,除非要等待多个事件或需跨越多个进程。

此外,用了一下发现,ManualResetEventSlim用的是Wait()而不是WaitOne(),两者比较大的区别在于Wait()还可以接收一个取消令牌。

AutoResetEvent

AutoResetEvent类表示线程同步事件在一个等待线程释放后收到信号时自动重置

注意,AutoResetEvent对象调用Set()后会自动调用Reset()方法,这也是类名的由来,而ManualResetEvent其功能跟AutoResetEvent是一样的,只是在调用Set()方法后不会自动调用Reset()方法。此外,AutoResetEvent只解除一个线程的Wait()调用所造成的阻塞,使用自动重置事件,很容易在编写生产者线程时发生失误,导致它的迭代次数多于消费者线程。

实际上,应该尽量避免使用AutoResetEvent,而选择使用ManualResetEventManualResetEventSlim

  • 官方示例

using System;
using System.Threading;// Visual Studio: Replace the default class in a Console project with 
//                the following class.
class Example
{private static AutoResetEvent event_1 = new AutoResetEvent(true);private static AutoResetEvent event_2 = new AutoResetEvent(false);static void Main(){Console.WriteLine("Press Enter to create three threads and start them.\r\n" +"The threads wait on AutoResetEvent #1, which was created\r\n" +"in the signaled state, so the first thread is released.\r\n" +"This puts AutoResetEvent #1 into the unsignaled state.");Console.ReadLine();for (int i = 1; i < 4; i++){Thread t = new Thread(ThreadProc);t.Name = "Thread_" + i;t.Start();}Thread.Sleep(250);for (int i = 0; i < 2; i++){Console.WriteLine("Press Enter to release another thread.");Console.ReadLine();event_1.Set();Thread.Sleep(250);}Console.WriteLine("\r\nAll threads are now waiting on AutoResetEvent #2.");for (int i = 0; i < 3; i++){Console.WriteLine("Press Enter to release a thread.");Console.ReadLine();event_2.Set();Thread.Sleep(250);}// Visual Studio: Uncomment the following line.//Console.Readline();}static void ThreadProc(){string name = Thread.CurrentThread.Name;Console.WriteLine("{0} waits on AutoResetEvent #1.", name);event_1.WaitOne();Console.WriteLine("{0} is released from AutoResetEvent #1.", name);Console.WriteLine("{0} waits on AutoResetEvent #2.", name);event_2.WaitOne();Console.WriteLine("{0} is released from AutoResetEvent #2.", name);Console.WriteLine("{0} ends.", name);}
}
  • 实际用例
    下面是下载文件的一部分代码,遍历文件列表,每一个文件开始下载后就阻塞当前线程,直到文件下载完成后,再释放线程。
private AutoResetEvent autoResetEvent = new AutoResetEvent(false);
private void OnStart()
{Task.Run(() =>{Files.ToList().ForEach(f =>{WebAccess webAccess = new WebAccess();webAccess.DownloadProgressChanged = (int progressPercent) =>{Progress = progressPercent; //设置进度泡泡};webAccess.DownLoadCompleted = () =>{autoResetEvent.Set();};webAccess.Download(f, $@".\{f.FileName}");autoResetEvent.WaitOne();});});
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/673832.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

uniapp 开发App 权限授权 js-sdk

从官网的插件市场下载的&#xff1a; 直接上代码&#xff1a; /*** 本模块封装了Android、iOS的应用权限判断、打开应用权限设置界面、以及位置系统服务是否开启*/var isIos // #ifdef APP-PLUS isIos (plus.os.name "iOS") // #endif// 判断推送权限是否开启 fu…

Excel VBA exit退出循环

Excel VBA exit退出循环 exit for退出for循环&#xff0c;exit do用于退出do 或 do while循环&#xff1a; 获取非空行 Function GetNotNullRow(ByVal iStartRow, ByRef iRow) Dim Rng As Range For Each Rng In Range(“B” & iStartRow & “:” & “P” & …

【开源】JAVA+Vue+SpringBoot实现公司货物订单管理系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 客户管理模块2.2 商品维护模块2.3 供应商管理模块2.4 订单管理模块 三、系统展示四、核心代码4.1 查询供应商信息4.2 新增商品信息4.3 查询客户信息4.4 新增订单信息4.5 添加跟进子订单 五、免责说明 一、摘要 1.1 项目…

微信小程序新手入门教程四:样式设计

WXSS (WeiXin Style Sheets)是一套样式语言&#xff0c;用于描述 WXML 的组件样式&#xff0c;决定了 WXML 的组件会怎么显示。 WXSS 具有 CSS 大部分特性&#xff0c;同时为了更适合开发微信小程序&#xff0c;WXSS 对 CSS 进行了扩充以及修改。与 CSS 相比&#xff0c;WXSS …

Tkinter教程21:Listbox列表框+OptionMenu选项菜单+Combobox下拉列表框控件的使用+绑定事件

------------★Tkinter系列教程★------------ Tkinter教程21&#xff1a;Listbox列表框OptionMenu选项菜单Combobox下拉列表框控件的使用绑定事件 Tkinter教程20&#xff1a;treeview树视图组件&#xff0c;表格数据的插入与表头排序 Python教程57&#xff1a;tkinter中如何…

Flink Format系列(2)-CSV

Flink的csv格式支持读和写csv格式的数据&#xff0c;只需要指定 format csv&#xff0c;下面以kafka为例。 CREATE TABLE user_behavior (user_id BIGINT,item_id BIGINT,category_id BIGINT,behavior STRING,ts TIMESTAMP(3) ) WITH (connector kafka,topic user_behavior…

Python中的权限控制

面向对象的三大特征&#xff1a; 1.封装&#xff1a;隐藏内部细节&#xff0c;对外提供操作方式。 权限控制&#xff1a;是通过对属性或方法添加单下划线、双下划线以及首尾下划线来实现 单下划线开头&#xff1a;以单下划线开头的属性或方法表示protected受保护的成员&…

GPT4_VS_ChatGPT(from_nytimes)

GPT4 VS ChatGPT&#xff08;from nytimes &#xff09; 正如文章官网博文&#xff1a;https://openai.com/research/gpt-4所述&#xff0c;GPT4仍有很多不足之处&#xff0c;还不及人类水平。纽约时报报道了一些人体验GPT4的效果和一些评价&#xff1a; Cade Metz 要求专家使…

什么是制动电阻器?工作及其应用

电梯、风力涡轮机、起重机、升降机和电力机车的速度控制是非常必要的。因此&#xff0c;制动电阻器是这些应用不可或缺的一部分&#xff0c;因为它们是电动机驱动器中最常用的高功率电阻器&#xff0c;用于控制其速度&#xff0c;在运输、海事和建筑等行业中。 电动火车主要比柴…

navigator.mediaDevices.getUserMedia获取本地音频/麦克权限并提示用户

navigator.mediaDevices.getUserMedia获取本地音频/麦克权限并提示用户 效果获取权限NotFoundErrorNotAllowedError 代码 效果 获取权限 NotFoundError NotAllowedError 代码 // 调用 captureLocalMedia()// 方法 function captureLocalMedia() {console.warn(Requesting lo…

redis特点

一、redis线程模型有哪些&#xff0c;单线程为什么快&#xff1f; 1、IO模型维度的特征 IO模型使用了多路复用器&#xff0c;在linux系统中使用的是EPOLL 类似netty的BOSS,WORKER使用一个EventLoopGroup(threads1) 单线程的Reactor模型&#xff0c;每次循环取socket中的命令…

oracle 启动命令以及ORA-01033问题处理、删除归档日志

1 启动数据库:startup 2 关闭数据库&#xff1a;Shutdown immediate 3 查看监听状态&#xff1a;lsnrctl status 4 启动监听&#xff1a;lsnrctl start 5 停止监听&#xff1a;lsnrctl stop 常见问题 1、在服务器重启后会出现&#xff0c;Oracle ORA-01033: ORAC…

Java线程是怎么实现run方法的执行的呢?【 多线程在JVM中的实现原理剖析】

Java线程是怎么实现run方法的执行的呢&#xff1f;【 多线程在JVM中的实现原理剖析】 查看naive state0 方法JVM_StartThread 方法创建操作系统线程操作系统线程执行 本文转载-极客时间 我们知道Java线程是通过行start()方法来启动的&#xff0c;线程启动后会执行run方法内的代…

数据可视化教程!我将全程出镜解说

你好&#xff0c;我是郭震 数据和算力是助推AI成功的关键二要素&#xff0c;洞察数据背后奥秘的重要手段之一便是数据可视化。 数据可视化无论在论文发表&#xff0c;还是商业数据分析中都扮演着举足轻重的作用。 一句话&#xff1a;一图胜千言。 从今天开始&#xff0c;我会逐…

FATFS学习笔记——FATFS写文件的两种方式

1.以hex文件直接写&#xff1a; 适用于需要保存比较大的数据时,无需进行格式化处理&#xff0c;直接保存文件 FIL file;FRESULT res;UINT bytes_written;// 创建文件res f_open(&file, "0:/FATFS/TestData.txt", FA_CREATE_ALWAYS | FA_WRITE);if (res ! FR_OK…

【Script】使用pyOpenAnnotate搭建半自动标注工具(附python源码)

文章目录 0. Background1. Method2. Code3. Example: 雄鹿红外图像标注3.1 选择色彩空间3.2 执行阈值3.3 执行形态学操作3.4 轮廓分析以找到边界框3.5 过滤不需要的轮廓3.6 绘制边界框3.7 以需要的格式保存Reference本文将手把手教你用Python和OpenCV搭建一个半自动标注工具(包…

【项目源码】一套基于springboot+Uniapp框架开发的智慧医院3D人体导诊系统源码

智慧医院3D人体导诊系统源码 开发语言:java 开发工具:IDEA 前端框架:Uniapp 后端框架:springboot 数 据 库:mysql 移 动 端:微信小程序、H5 “智慧导诊”以人工智能手段为依托,为人们提供智能分诊、问病信息等服务,在一定程度上满足了人们自我健康管理、精准挂号…

6个好看的wordpress模板

简站wordpress服务业通用主题 2023年立秋纪念版&#xff0c;简站wordpress服务行业通用主题&#xff0c;适合服务行业企业官网使用。 https://www.jianzhanpress.com/?p5393 小语种翻译wordpress主题 小语种国家外贸网站建设需要的wordpress主题模板&#xff0c;适合做小语…

单链表实现约瑟夫环

大家对约瑟夫环是比较陌生的&#xff0c;但是对于大多数人来说&#xff0c;丢手绢却一点都不陌生&#xff0c;其实约瑟夫环和丢手绢差不多。 约瑟夫环 约瑟夫环(约瑟夫问题)是一个数学的应用问题:已知n个人(以编号1&#xff0c;2&#xff0c;3…n分别表示)围坐在一张圆桌周围。…

Android 10.0 系统sim卡4G LTE信号格数对应的信号强度值修改

1.前言 在10.0的系统产品开发中,在手机通讯这块功能也是特别重要的,在手机插入手机sim卡后,会根据当前的sim卡信号来显示对应的 sim卡信号显示格数,所以为了增强相关的信号格数,就需要修改sim卡对应的信号强度值来实现相关的功能 2.系统sim卡4G LTE信号格数对应的信号强…