C# .Net学习笔记—— 异步和多线程(异常处理)

一、异常处理

1、下面for循环20个线程,到11,12号的时候执行失败,这里我也用了try catch来捕获异常。

 private void button11_Click(object sender, EventArgs e)
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();
            try
            {
                for (int i = 0; i < 20; i++)
                {
                    string name = string.Format($"Click_{i}");
                    Action<object> act = t =>
                    {
                        Thread.Sleep(2000);
                        if (t.ToString().Equals($"Click_11"))
                        {
                            throw new Exception(string.Format($"{t} 执行失败"));
                        }
                        if (t.ToString().Equals($"Click_12"))
                        {
                            throw new Exception(string.Format($"{t} 执行失败"));
                        }
                        Console.WriteLine("{0} 执行成功", t);
                    };
                    taskList.Add(taskFactory.StartNew(act, name));
                }
            }
            catch (AggregateException aex)
            {
                foreach (var item in aex.InnerExceptions)
                {
                    Console.WriteLine(item.Message);
                }
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex);
            }
        }

打印出来发现并没有捕获到异常

那么我再新增一句:

Task.WaitAll(taskList.ToArray());

这样我们就可以成功捕获到异常了。 同时,我们也可以通过AggregateException,捕获到我们异常的数据。

 最开始我们抓不到异常,是因为系统跑出了try catch,我们抓不到。

WaitAll可以抓到多线程里面的所有异常

但是产生了一个新的问题,WaitAll会卡界面

线程里面的action不允许出现异常,需要自己处理好

二、线程取消

多个线程并发,某个失败后,希望别的线程停下来。

task外部无法中止,Thread.Abort不靠谱,因为线程是OS的资源,无法掌控啥时候取消

线程自己停止自己——公共的访问变量——修改它——线程不断的检测它(会有延迟)

CancellationTokenSource标志任务是否取消    Cancel 表示取消  IsCancellationRequested表示是否取消。

Token启动Task的时候传入,那么如果Cancel 了,这个任务就会放弃启动,抛出一个异常

private void button12_Click(object sender, EventArgs e)
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();
            CancellationTokenSource cts = new CancellationTokenSource();  //bool值
            for (int i = 0; i < 20; i++)
            {
                string name = $"Click_{i}";
                Action<object> act = t =>
                {
                    try
                    {
                        Thread.Sleep(2000);
                        if (t.ToString().Equals($"Click_11"))
                        {
                            throw new Exception(string.Format($"{t} 执行失败"));
                        }
                        if (t.ToString().Equals($"Click_12"))
                        {
                            throw new Exception(string.Format($"{t} 执行失败"));
                        }
                        //除了11和12抛异常,其他的我们检查一下。
                        //如果已经取消了,那么我们放弃执行
                        if (cts.IsCancellationRequested) //检查信号量
                        {
                            Console.WriteLine($"{t} 放弃执行");
                            return;
                        }
                        //如果还没取消了,那么我们正确执行
                        else
                        {
                            Console.WriteLine($"{t} 执行成功");
                        }
                    }
                    catch (Exception ex)
                    {
                        cts.Cancel();
                        Console.WriteLine(ex.Message);
                    }
                };
                taskList.Add(taskFactory.StartNew(act, name, cts.Token));
            }
            Task.WaitAll(taskList.ToArray());
        }

 三、多线程的临时变量

1、闭包问题

       private void button13_Click(object sender, EventArgs e)
        {
            for (int i = 0; i < 5; i++) 
            {
                int k = i;
                Task.Run(() => 
                {
                    Thread.Sleep(100);
                    Console.WriteLine(k);
                });
            }
        }

 四、多线程的线程安全问题

抛出问题:

private void button14_Click(object sender, EventArgs e)
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();
            int totalCount = 0;
            List<int> intList = new List<int>();
            for (int i = 0; i < 10000; i++)
            {
                int newi = i;
                taskList.Add(taskFactory.StartNew(() => 
                {
                    totalCount += 1;
                    intList.Add(newi);
                }));
            }
            Task.WaitAll(taskList.ToArray());
            Console.WriteLine(totalCount);
            Console.WriteLine(intList.Count);
        }

这里我声明了一个int类型的临时变量和一个List<int>类型的临时变量,让他们嵌套在多线程内进行累加。

 可以发现打印出来的数目并不一致,并且都小于10000

这就是线程安全的问题,是多个线程同时操作同一变量导致的。

对于共有变量:都能访问的局部变量/全局变量/数据库的值/硬盘文件

 我们尝试加上锁(Lock)(Lock可以算是一种语法糖)。

 private static readonly object btnThreadCore_Click_Lock = new object();
        private void button14_Click(object sender, EventArgs e)
        {
            TaskFactory taskFactory = new TaskFactory();
            List<Task> taskList = new List<Task>();

            int totalCount = 0;
            List<int> intList = new List<int>();
            for (int i = 0; i < 10000; i++)
            {
                int newi = i;
                taskList.Add(taskFactory.StartNew(() => 
                {
                    lock (btnThreadCore_Click_Lock)
                    {
                        totalCount += 1;
                        intList.Add(newi);
                    }
                }));
            }
            Task.WaitAll(taskList.ToArray());
            Console.WriteLine(totalCount);
            Console.WriteLine(intList.Count);
        }

发现打印出来的数量都正常。

因为lock后的方法块,任意时刻只有一个线程可以进入

1、Lock介绍:

介绍:Lock等同于Monitor.Enter(btnThreadCore_Click_Lock); 

        离开等同于调用了Monitor.Exit();

限制:Lock只能锁引用类型(占用引用链接),不要用string,因为享元

        微软提供的标准写法:

        private static readonly object btnThreadCore_Click_Lock = new object();

        Lock 最好锁private的,防止外面也去lock

        static 全场唯一 ,避免不同实例锁的不同

        readonly 只读,不要改动

        object 表示引用

lock(this)每次实例化都是不同的锁,同一个实例时相同的锁

但是这个实例别人也能访问到,别人也能锁定

缺点:

 Lock解决,因为只有一个线程可以进去,没有并发,所以解决了问题,但是牺牲了性能,所以要尽量缩小lock的范围

2、安全队列(ConcurrentQueue):

最后还是需要一个线程完成操作

3、最好的方法就是不要冲突——数据拆分,避免冲突!!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

湖仓架构的演进

1.数据仓库架构的历史演进 起初&#xff0c;业界数据处理首选方式是数仓架构。通常数据处理的流程是把一些业务数据库&#xff0c;通过ETL的方式加载到Data Warehouse中&#xff0c;再在前端接入一些报表或者BI的工具去展示。 数据仓库概念是 Inmon 于 1990 年提出并给出了完…

文献综述方法论|全文翻译

最常见的错误是文献综述往往未能为该领域提供真正有价值的贡献。无论综述文章多么优秀和严谨&#xff0c;如果它没有提供足够的新内容&#xff0c;就不会被发表。太常见的情况是&#xff0c;文献综述只是对特定年份之间进行的研究进行描述性总结&#xff0c;描述了诸如发表的文…

聚会小游戏+摇色子+愤怒的大叔+真心话太冒险微信小程序源码系统:活跃气氛神器 带完整的安装包以及搭建教程

在现代社交活动中&#xff0c;如何快速破冰并调动气氛一直是人们关注的焦点。微信小程序以其便捷性、互动性和多样性成为了解决这一问题的理想工具。今天&#xff0c;小编将为大家介绍一款集聚会小游戏、摇色子、真心话大冒险等功能于一身的微信小程序源码系统——“活跃气氛神…

Leetcode13-解密消息(2325)

1、题目 给你字符串 key 和 message &#xff0c;分别表示一个加密密钥和一段加密消息。解密 message 的步骤如下&#xff1a; 使用 key 中 26 个英文小写字母第一次出现的顺序作为替换表中的字母 顺序 。 将替换表与普通英文字母表对齐&#xff0c;形成对照表。 按照对照表 …

[C#]使用OpenCvSharp实现区域文字提取

【官方框架地址】 github.com/shimat/opencvsharp 【算法介绍】 采用opencv算法实现文字区域提取&#xff0c;步骤如下&#xff1a; &#xff08;1&#xff09;形态学操作 &#xff08;2&#xff09;查找轮廓 &#xff08;3&#xff09;筛选那些面积小的 &#xff08;4&#…

Element ui 改变el-transfer 穿梭框的大小

修改el-transfer 左右两个穿梭框的高度和宽度&#xff0c;具体效果如下正常大小的穿梭框修改之后的&#xff0c;主要在style中加上如下样式即可 /deep/ .el-transfer-panel{ width: 470px; /* 左右两个穿梭框的高度和宽度 */ height: 450px; } /deep/ .el-transfer-panel__li…

【Bootstrap5学习 day10】

Flex布局 弹性盒子是CSS3的一种新的布局模式&#xff0c;更适合响应式的设计 创建一个弹性盒子容器 使用d-flex类&#xff0c;创建flexbox容器并将直接子项转换为flex项 <div class"d-flex p-3 bg-info text-white"><div class"p-2 bg-secondary"…

03Spring实现IoC:依赖注入/构造注入

● 控制反转&#xff0c;反转的是什么&#xff1f; ○ 将对象的创建权利交出去&#xff0c;交给第三方容器负责。 ○ 将对象和对象之间关系的维护权交出去&#xff0c;交给第三方容器负责。 ● 控制反转这种思想如何实现呢&#xff1f; ○ DI&#xff08;Dependency Injection&…

G1为什么更适合亿级流量系统以及YGC优化策略screenflow

大白话&#xff1a; 1.ParNew执行回收的时候&#xff0c;STW会比较长&#xff0c;CMS存在碎片化的问题&#xff0c;当物理机的内存变大&#xff0c;这套组合存在的问题会更大&#xff0c;加大物理内存&#xff0c;反而让垃圾回收更慢。 大白话&#xff1a; 之前讲过&#xff0c…

vue-打包

打包的作用 说明&#xff1a;vue脚手架只是开发过程中&#xff0c;协助开发的工具&#xff0c;当真正开发完了>脚手架不参与上线 打包的作用&#xff1a; 1&#xff09;将多个文件压缩合并成一个文件 2&#xff09;语法降级 3&#xff09;less sass ts语法解析 打包后…

羊大师解读,羊奶的口味更适合哪些人群?

羊大师解读&#xff0c;羊奶的口味更适合哪些人群&#xff1f; 羊奶作为一种营养丰富的乳制品&#xff0c;拥有许多独特的品质和口味&#xff0c;备受消费者的青睐。它不仅含有丰富的蛋白质、维生素和矿物质&#xff0c;还具有更易消化的特点&#xff0c;适合许多人群的饮用。…

CSS新增文本描边-text-stroke属性

-webkit-text-stroke属性 概念&#xff1a;-webkit-text-stroke属性为文本添加描边效果。所谓的描边效果&#xff0c;指的是给文字添加边框 语法&#xff1a; -webkit-text-stroke:width color;Chrome和Firefox这两个浏览器都只能识别带有-webkit前缀的text-stroke属性 -web…

【HarmonyOS开发】ArkUI-X 跨平台框架(使用ArkTs开发AndroidIOS)

ArkUI-X 跨平台框架进一步将 ArkUI 开发框架扩展到了多个OS平台&#xff0c;目前支持OpenHarmony、HarmonyOS、Android、 iOS&#xff0c;后续会逐步增加更多平台支持。开发者基于一套主代码&#xff0c;就可以构建支持多平台的精美、高性能应用。 一、跨平台框架有哪些? 1、…

CyberLink的视频编辑软件PowerDirector Ultimate 2024 22.0版本在win系统下载与安装配置

目录 前言一、PowerDirector Ultimate安装二、使用配置总结 前言 PowerDirector Ultimate是由CyberLink公司开发的一款视频编辑软件&#xff0c;其为高级版本&#xff0c;拥有多种强大的视频编辑和效果功能。该软件具有许多强大的功能和工具&#xff0c;包括多轨时间线编辑、视…

DevEco Studio集成ArkUI-X

DevEco StudioHarmonyOs教程 &#xff08;免费学&#xff09;&#xff1a; 最新HarmonyOS系列教程下载地址-IT营大地老师--更新中 ArkUI-X进一步将ArkUI扩展到了多个OS平台&#xff1a;目前支持OpenHarmony、HarmonyOS、Android、 iOS&#xff0c;后续会逐步增加更多平台支持。…

手写一个加盐加密算法(java实现)

目录 前言 什么是MD5&#xff1f;&#xff1f; 加盐算法 那别的人会不会跟你得到相同的UUID&#xff1f; 如何使用盐加密&#xff1f; 代码实现 前言 对于我们常见的登录的时候需要用到的组件&#xff0c;加密是一个必不可少的东西&#xff0c;如果我们往数据库存放用户…

怎么查询网络出口IP

怎么查询自己的网络的出口IP 背景 一般跟第三方服务进行接口数据交互的时候&#xff0c;对方都会让我们提供调用接口的网络的出口IP&#xff0c;对方会把该IP地址加到对方的白名单中。这样我们才能有权限进行接口的访问。 解决办法 下面介绍三种常用的查询网络出口IP的办法…

弧垂观测手段再升级!输电线路导线弧垂检测装置的应用_深圳鼎信

输电线路导线弧垂是指在输电线路中导线的水平位置与塔杆之间的垂直距离。导线的弧垂是确定导线张力、塔杆高度等参数的重要依据。通过测量弧垂及时调整弧垂大小对保证输电线路的安全运行具有重要作用。鼎信将介绍两种测量弧垂的方法&#xff0c;一起来学习一下吧&#xff01; …

设计模式篇章(2)——五种创建者模式

创建者模式主要思考如何创建一个对象&#xff0c;如何将对象的创建与使用分离。一般初级程序员都是new一个对象&#xff0c;然后紧接着使用这个对象&#xff0c;在某些场景中这样子是有问题的&#xff0c;需要使用创建者模式替代的&#xff08;例如使用单例模式&#xff09;。设…

第G2周:人脸图像生成(DCGAN)

&#x1f368; 本文为[&#x1f517;365天深度学习训练营学习记录博客\n&#x1f366; 参考文章&#xff1a;365天深度学习训练营\n&#x1f356; 原作者&#xff1a;[K同学啊 | 接辅导、项目定制]\n&#x1f680; 文章来源&#xff1a;[K同学的学习圈子](https://www.yuque.co…