ManualResetEvent实现线程的暂停与恢复

背景

前些天遇到一个需求,在没有第三方源码的情况下,刷新一个第三方UI,并且拦截到其ajax请求的返回结果。当结果为AVALIABLE的时候,停止刷新并语音提示,否则继续刷新。

分析这个需求,发现需要控制一个刷新循环的暂停与开始,因此网上搜到了通过ManualResetEvent实现线程的暂停与恢复。

ManualResetEvent介绍

ManualResetEvent是一个通过信号机制,实现线程间状态同步的类。常用的方法有以下三个:

WaitOne:阻止当前线程,直到收到信号

Reset:将事件状态设置为非终止状态,导致线程阻止

Set:将事件状态设置为终止状态,从而允许继续执行一个或多个等待线程

实现设计

1.通过CefSharp的谷歌浏览器插件请求第三方网站

41aaa78520e708e1af3049f352e112aa.png

 2.具体交互逻辑如下

  1. 默认开启一个线程,并通过WaitOne挂起,等待手动开始自动刷新的指令

  2. 手动通过Set方法发送开始工作信号

  3. 执行完面操作后,通过Reset将线程再次挂起,并等待Ajax结果,判断Ajax结果,如果不等于AVALIABLE则再次恢复线程

public Form1()
{InitializeComponent();var setting = new CefSettings();setting.Locale = "zh-CN";setting.UserAgent = "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/50.0.2661.102 Safari/537.36";// 避免页面加载不出来, 建议加上这句Cef.Initialize(setting);workerThread = new Thread(new ThreadStart(BeginListenWorkerAsync));workerThread.IsBackground = true;workerThread.Start();this.停止自动监听ToolStripMenuItem.Enabled = false;
}private async void BeginListenWorkerAsync()
{while (true){_eventBeginListenWorkList.WaitOne();  //1.默认挂起线程,等待信号chromiumWebBrowser1.GetBrowser().MainFrame.ExecuteJavaScriptAsync(@"document.evaluate(""//*[@id='root']/div[1]/div[2]/div/div[2]/div[1]/div[1]/div/div[1]/button"", document).iterateNext().click()");Thread.Sleep(1000);chromiumWebBrowser1.GetBrowser().MainFrame.ExecuteJavaScriptAsync(@"document.evaluate(""//*[@id='root']/div[1]/div[2]/div/div[1]/div/div/div/div[2]/div[6]/div/button"", document).iterateNext().click()");_eventBeginListenWorkList.Reset(); //3.页面执行完相应操作则再次挂起线程,等待ajax请求结果再判断是否恢复线程}
}private void 停止自动监听ToolStripMenuItem_Click(object sender, EventArgs e)
{this.开始自动监听ToolStripMenuItem.Enabled = true;this.停止自动监听ToolStripMenuItem.Enabled = false;_eventBeginListenWorkList.Reset();StartListen = false;
}private void 开始自动监听ToolStripMenuItem_Click(object sender, EventArgs e)
{this.开始自动监听ToolStripMenuItem.Enabled = false;this.停止自动监听ToolStripMenuItem.Enabled = true;_eventBeginListenWorkList.Set(); // 2.手动通过Set发送信号恢复线程,让其工作StartListen = true;
}

 3.现在我们需要定义RequestHandler来指定ResourceHandler拦截ajax请求

private void Form1_Load(object sender, EventArgs e)
{chromiumWebBrowser1.RequestHandler = new MyRequestHandler(this);chromiumWebBrowser1.Load("https://www.xxxxxxxx.com/");
}

MyRequestHandler中需要指定自定义ResourceHandler

public class MyRequestHandler : RequestHandler
{Form1 _form;public MyRequestHandler(Form1 form){_form = form;}public static string AuthorizationValue;protected override bool OnBeforeBrowse(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame, IRequest request, bool userGesture,bool isRedirect){// 先调用基类的实现,断点调试return base.OnBeforeBrowse(chromiumWebBrowser, browser, frame, request, userGesture, isRedirect);}protected override IResourceRequestHandler GetResourceRequestHandler(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame,IRequest request, bool isNavigation, bool isDownload, string requestInitiator, ref bool disableDefaultHandling){Console.WriteLine(request.Url);if (request.Url.StartsWith("https://www.xxxxxxxxxxxxxxxx/api") && request.Headers.AllKeys.Contains("Authorization")){AuthorizationValue = request.Headers["Authorization"];}return new MyResourceRequestHandler(_form);}
}

4.最后ResourceRequestHandler拦截到ajax请求之后,如果结果不是AVALIABLE则通过StartListenFunc再次恢复线程

public class MyResourceRequestHandler : ResourceRequestHandler
{Form1 _form;public MyResourceRequestHandler(Form1 form){_form = form;}private Dictionary<ulong, MemoryStreamResponseFilter> responseDictionary = new Dictionary<ulong, MemoryStreamResponseFilter>();protected override IResponseFilter GetResourceResponseFilter(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IResponse response){var dataFilter = new MemoryStreamResponseFilter();responseDictionary.Add(request.Identifier, dataFilter);return dataFilter;}protected override void OnResourceLoadComplete(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IResponse response, UrlRequestStatus status, long receivedContentLength){MemoryStreamResponseFilter filter;if (responseDictionary.TryGetValue(request.Identifier, out filter)){var data = filter.Data;if (request.Url.StartsWith("https://xxxxxxxxxxxxxxxxxxxx/oneapi")){var available = false;string s = System.Text.Encoding.UTF8.GetString(data, 0, data.Length);Console.WriteLine(s);if (!string.IsNullOrEmpty(s)){var responseData = JsonConvert.DeserializeObject<ScheduleResponse>(s);if (responseData != null && responseData.productPreviews != null){foreach (var item in responseData.productPreviews){if (item.schedule != null && item.schedule.status != null && item.schedule.status.name == "AVAILABLE"){available = true;Mp3Player.Play();Action act = delegate () { _form.StopListen(); };_form.Invoke(act);break;}}}}if (!available && Form1.StartListen){Form1.StartListenFunc();}}filter.Dispose();}}
}

 而StartListenFunc则是通过Set方法再次恢复了线程对ui的重复查询

public static void StartListenFunc()
{_eventBeginListenWorkList.Set();StartListen = true;
}

ManualResetEvent官方介绍如下

https://docs.microsoft.com/zh-cn/dotnet/api/system.threading.manualresetevent?view=netframework-1.1

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

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

相关文章

浙大哈佛剑桥学者联手破解数学界几十年的谜题,成果登上数学顶刊

全世界只有3.14 % 的人关注了爆炸吧知识转自&#xff1a;量子位作者&#xff1a;边策 萧箫当两个看似“无关”的数学领域发生碰撞&#xff0c;会发生什么&#xff1f;浙江大学研究员、中科大数学系2003级校友叶和溪&#xff0c;与来自剑桥大学、哈佛大学的两位学者一起&#xf…

Hibernate 参数设置一览表

Hibernate 参数设置一览表属性名用途hibernate.dialect一个Hibernate Dialect类名允许Hibernate针对特定的关系数据库生成优化的SQL. 取值 full.classname.of.Dialect hibernate.show_sql输出所有SQL语句到控制台. 有一个另外的选择是把org.hibernate.SQL这个log category设为d…

jsp将鼠标放到那显示信息

将下面的代码&#xff0c;粘贴复制到jsp<body></body>中<A href"#" οnmοuseοver"document.getElementById(aaa).style.display" οnmοuseοut"document.getElementById(aaa).style.displaynone">鼠标移动到这我就出来了&l…

Monkey测试实例

Monkey是Android中的一个命令行工具&#xff0c;可以运行在模拟器里或实际设备中。它向系统发送伪随机的用户事件流(如按键输入、触摸屏输入、手势输入等)&#xff0c;实现对正在开发的应用程序进行压力测试。Monkey测试是一种为了测试软件的稳定性、健壮性的快速有效的方法。1…

.NET Core Runtime vs .NET Framework Runtime

在我从 .NET Framework 到 .NET Core 的过渡期间&#xff0c;有一件事情让我很长时间都感到困惑&#xff0c;那就是 Runtime 运行时&#xff0c; 实际上 Runtime 在 .NET Framework 和 .NET Core 中具有不同的含义。.NET Framework 运行时的历史 当有人问我们&#xff0c;什么是…

“玻璃大王”曹德旺捐资100亿办大学!幼年失学的他要打造理工科研究型大学...

全世界只有3.14 % 的人关注了爆炸吧知识本文转自募格学术整理自&#xff1a;量子位&#xff08;作者贾浩楠、鱼羊&#xff09;、青塔等100亿投资&#xff0c;福建省迎来第一个“新型大学”。办学资金&#xff0c;来自生长于福建&#xff0c;在福建发家致富的“玻璃大王”曹德旺…

JAVA配置Tomcat

1.下载tomcat&#xff0c;我jdk是1.8的&#xff0c;网上查了一下&#xff0c;说要安装tomcat8及以上的tomcat 尝试点击&#xff0c;弹出&#xff0c; 2.配置环境 3.安装通过cmd安装 4.点击开启服务 5.输入localhost:8080查看 6.还可以通过tomcat8w.exe来启动窗口&#xff0c;开…

女孩看男孩VS男孩看女孩

1 这波猫粮我吃了&#xff01;2 论一只狗子的自娱自乐精神&#xff01;3 分享一个 “想死你了”的拥抱&#xff01;4 老师&#xff1a;下课铃不代表这节课上完了5 谁怕谁啊&#xff01;6 好像很有道理的样子7 将它们展开会是什么&#xff1f;你点的每个赞&#xff0c;我都认真当…

使用建造者模式创建模拟数据

前言在写测试用例时&#xff0c;我们经常需要创建模拟数据&#xff0c;在C#中常用的方式是使用nuget包Bogus。Bogus可以按照一定规则生成随机数据&#xff0c;示例代码如下&#xff1a;public class User {public string Name { get; set; }public int Age { get; set; }public…

JMeter学习(六)集合点

JMeter也有像LR中的集合点&#xff0c;本篇就来介绍下JMeter的集合点如何去实现。 JMeter里面的集合点通过添加定时器来完成。 注意&#xff1a;集合点的位置一定要在Sample之前. 集合点&#xff1a;简单来理解一下&#xff0c;虽然我们的“性能测试”理解为“多用户并发测试”…

无意间看到的浏览器记录......

1 喵&#xff1a;我是风一样的喵主子...啊错了&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼2 老了谁照顾你&#xff1f;▼3 帮妈妈清理手机垃圾▼4 你现在做什么工作&#xff1f;&#xff08;via.欢声笑语bot&#xff0c;侵删&#xff09;▼5 社会社会&#xf…

linux C/C++开发环境搭建指南

2019独角兽企业重金招聘Python工程师标准>>> Eclipse IDE for C/C developers 就是安装了CDT的Eclipse IDE了。 一、安装基本开发环境 本人想学习一下Linux下的C&#xff0c;C&#xff0b;&#xff0b;程序开发&#xff0c;这几天一直在研究Linux下的C语言编译环境…

网站能拿到其他网站的cookie_如何设计网站能让网站建设的更有吸引力

现在做互联网营销的企业基本都是有自己的企业网站&#xff0c;不过要做好企业网站不能简简单单的套个模板&#xff0c;这样做出来的网站毫无吸引力&#xff0c;做好的网站那么网站的设计的很重要的&#xff0c;换个思维大家想一下&#xff0c;网站的设计就和我们线下开门店装修…

.NET6之MiniAPI(九):基于角色的身份验证和授权

身份验证是这样一个过程&#xff1a;由用户提供凭据&#xff0c;然后将其与存储在操作系统、数据库、应用或资源中的凭据进行比较。 在授权过程中&#xff0c;如果凭据匹配&#xff0c;则用户身份验证成功&#xff0c;可执行已向其授权的操作。 授权指判断允许用户执行的操作的…

flex容器属性(一)

一&#xff0c;概念 flexible box ,意为“弹性布局”&#xff0c;用来为盒状模型提供最大的灵活性。 块级布局更侧重于垂直方向&#xff0c;行内布局更侧重于水平方向&#xff0c;于此相对的&#xff0c;弹性盒子布局算法是方向无关的。 块级flex布局&#xff1a; .box{display…

关于最近打的几题斜率优化的总结。加几AC代码。

斜率优化错误总结 网上说很多OJ桑的斜率优化大多都是模板题- -&#xff0c;结果每次都跪Orz。。。在此总结一些常见错误&#xff1a; 1&#xff1a;不得不说斜率优化很多时候计算式很长- -&#xff0c;代码容易错细节- -。 2&#xff1a;其次就是弹队头以及弹队尾的时候大小关系…

iio Engine logoHTML5 应用框架 iio Engine

iio Engine 是一个新的 HTML5 应用开源框架&#xff0c;基于 JavaScript 和 Canvas 开发&#xff0c;集成了 Box2D 在线演示&#xff1a;http://www.huiyi8.com/divcss/转载于:https://www.cnblogs.com/lhrs/p/4138106.html

带圈汉字 在线生成_手写签名在线生成器-手写签名在线生成器可复制

签名设计地址&#xff1a;www.mgs2s.com&#xff08;复制到浏览器打开&#xff09;工具集成签名设计免费版下载&#xff0c;签名设计免费版在线立即生成&#xff0c;简单简体签名设计免费版。最新方便设计公文签名设计颜色保存分享免费版1、所以大家最好写签名的时候&#xff0…

真快!10秒内将k8s集群运行起来

大家好&#xff0c;我是小碗汤&#xff0c;今天演示一个项目&#xff0c;可以在一分钟内用容器将k8s集群运行起来&#xff0c;真的很方便。您可能已经知道&#xff0c;将 Kubernetes 集群安装在 VM 上。但在 Docker 容器中安装一个 Kubernetes 集群&#xff0c;还没有太多的实践…