我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C#多线程相关操作的示例分析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C#多线程相关操作的示例分析

这篇文章主要介绍了C#多线程相关操作的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

一、线程异常

我们在单线程中,捕获异常可以使用try-catch,代码如下所示:

using System;namespace MultithreadingOption{    class Program    {        static void Main(string[] args)        {            #region 单线程中捕获异常            try            {                int[] array = { 1, 23, 61, 678, 23, 45 };                Console.WriteLine(array[6]);            }            catch (Exception ex)            {                Console.WriteLine($"message:{ex.Message}");            }            #endregion            Console.ReadKey();        }    }}

程序运行结果:

C#多线程相关操作的示例分析

那么在多线程中如何捕获异常呢?是不是也可以使用try-catch进行捕获?我们先看下面的代码:

using System;using System.Threading.Tasks;namespace MultithreadingOption{    class Program    {        static void Main(string[] args)        {            #region 单线程中捕获异常            //try            //{            //    int[] array = { 1, 23, 61, 678, 23, 45 };            //    Console.WriteLine(array[6]);            //}            //catch (Exception ex)            //{            //    Console.WriteLine($"message:{ex.Message}");            //}            #endregion            #region 多线程中的异常            try            {                for (int i = 0; i < 30; i++)                {                    string str = $"main_{i}";                    // 开启线程                    Task.Run(() =>                     {                        Console.WriteLine($"{str} 开始了");                        if(str.Equals("main_5"))                        {                            throw new Exception("main_5 发生了异常");                        }                        else if (str.Equals("main_11"))                        {                            throw new Exception("main_11 发生了异常");                        }                        else if (str.Equals("main_18"))                        {                            throw new Exception("main_18 发生了异常");                        }                        Console.WriteLine($"{str} 结束了");                    });                }            }            catch (Exception ex)            {                Console.WriteLine($"message:{ex.Message}");            }            #endregion            Console.ReadKey();        }    }}

程序运行结果:

C#多线程相关操作的示例分析

我们看到结果中并没有输出异常信息,是不是没有抛出异常呢?我们起代码进行调试,看调试信息:

C#多线程相关操作的示例分析

我们看到程序中确实也抛出了异常,但是程序却没有捕获到,那么异常去哪里了呢?异常被多线程给吞掉了,那么如何在多线程中捕获异常呢?如果把try-catch写在线程里面呢?每一个线程都是单线程的,把try-catch写在每一个线程里面就没有意义了。在多线程中捕获异常,需要使用到WaitAll(),看下面的代码:

try{     // 定义一个Task类型的List集合     List<Task> taskList = new List<Task>();     for (int i = 0; i < 30; i++)     {            string str = $"main_{i}";            // 开启线程,并把线程添加到集合中            taskList.Add(Task.Run(() =>            {                 Console.WriteLine($"{str} 开始了");                 if (str.Equals("main_5"))                 {                     throw new Exception("main_5 发生了异常");                 }                 else if (str.Equals("main_11"))                 {                      throw new Exception("main_11 发生了异常");                 }                 else if (str.Equals("main_18"))                 {                       throw new Exception("main_18 发生了异常");                 }                 Console.WriteLine($"{str} 结束了");          }));      }     // 等待所有线程都执行完     Task.WaitAll(taskList.ToArray());}catch (Exception ex){      Console.WriteLine($"message:{ex.Message}");}

我们用代码进行调试,调试结果:

C#多线程相关操作的示例分析

这时就可以进入到catch里面了,我们监视ex,发现ex是AggregateException类型的异常,我们在进一步优化代码:

try{     // 定义一个Task类型的List集合     List<Task> taskList = new List<Task>();     for (int i = 0; i < 30; i++)     {            string str = $"main_{i}";            // 开启线程,并把线程添加到集合中            taskList.Add(Task.Run(() =>            {                 Console.WriteLine($"{str} 开始了");                 if (str.Equals("main_5"))                 {                     throw new Exception("main_5 发生了异常");                 }                 else if (str.Equals("main_11"))                 {                      throw new Exception("main_11 发生了异常");                 }                 else if (str.Equals("main_18"))                 {                       throw new Exception("main_18 发生了异常");                 }                 Console.WriteLine($"{str} 结束了");          }));      }     // 等待所有线程都执行完     Task.WaitAll(taskList.ToArray());}catch(AggregateException are){     foreach (var exception in are.InnerExceptions)     {          Console.WriteLine(exception.Message);     }}catch (Exception ex){      Console.WriteLine($"message:{ex.Message}");}

最后运行程序:

C#多线程相关操作的示例分析

我们发现这时就可以捕获到具体的异常信息了。

二、线程取消

在上面的示例中,我们捕获到了多线程中发生的异常,并且也输出了异常信息,但是这样是不友好的。在实际开发中,我们使用多线程并发执行任务,假如其中某一个任务失败了或者发生了异常,我们希望可以通知其他的线程,都停止下来,那么该如何做呢?这时就需要使用到线程取消。

Task不能外部终止任务,只能自己终止自己。

.Net框架提供了CancellationTokenSource类,该类里面有一个bool类型的属性:IsCancellationRequested,默认是false,表示是否取消线程。还提供了一个Cancel()方法,该方法可以把IsCancellationRequested的属性值设置为true,并且不能在设置回去。代码如下:

// 实例化对象CancellationTokenSource cts = new CancellationTokenSource();for (int i = 0; i < 20; i++){      string str = $"main_{i}";      // 开启线程      Task.Run(() =>      {             try             {                  Console.WriteLine($"{str} 开始了");                  // 暂停                  Thread.Sleep(new Random().Next(50, 100) * 100);                  if (str.Equals("main_5"))                  {                       throw new Exception("main_5 发生了异常");                  }                  else if (str.Equals("main_11"))                  {                        throw new Exception("main_11 发生了异常");                  }                  if (cts.IsCancellationRequested == false)                  {                        Console.WriteLine($"{str} 结束了");                  }                  else                  {                         Console.WriteLine($"{str} 线程取消");                  }            }            catch (Exception ex)            {                   // 发生了异常,将IsCancellationRequested的值设置为true                   cts.Cancel();                   Console.WriteLine($"message:{ex.Message}");            }     });}

程序运行结果:

C#多线程相关操作的示例分析

可以看到,当有异常发生之后,有的线程就被取消了。这样就初步实现了线程取消。

在上面的示例中,我们是先开启了线程,如果发生了异常,则取消线程。那么会有这样一种情况:线程中发生了异常,可能这时候有的线程还没有开启,那么能不能就不让这些线程在开启呢?Task的Run方法有一个重载:

C#多线程相关操作的示例分析

第二个参数就表示取消线程。而且CancellationTokenSource类里面正好有这个参数:

C#多线程相关操作的示例分析

所以,我们可以利用Run方法的重载来实现不开启线程,代码如下:

try{    // 实例化对象    CancellationTokenSource cts = new CancellationTokenSource();    // 创建Task类型的集合    List<Task> taskList = new List<Task>();    for (int i = 0; i < 20; i++)    {        string str = $"main_{i}";        // 开启线程 Task.run 以后 添加Token 就可以在某一个线程发生异常之后,让没有开启的线程不开启了        taskList.Add(Task.Run(() =>        {            try            {                Console.WriteLine($"{str} 开始了");                // 暂停                Thread.Sleep(new Random().Next(50, 100) * 10);                if (str.Equals("main_5"))                {                    throw new Exception("main_5 发生了异常");                }                else if (str.Equals("main_11"))                {                    throw new Exception("main_11 发生了异常");                }                if (cts.IsCancellationRequested == false)                {                    Console.WriteLine($"{str} 结束了");                }                else                {                    Console.WriteLine($"{str} 线程取消");                }            }            catch (Exception ex)            {                // 发生了异常,将IsCancellationRequested的值设置为true                cts.Cancel();            }        }, cts.Token));    }    // 等待所有线程执行完    Task.WaitAll(taskList.ToArray());}catch (AggregateException are){    foreach (var exception in are.InnerExceptions)    {        Console.WriteLine(exception.Message);    }}

程序运行结果:

C#多线程相关操作的示例分析

输出结果中有一句话:已取消一个任务,但是我们的代码里面没有打印这句话,这是从哪里来的呢?这是因为第二个参数Token的原因,加了这个参数以后,如果就线程发生了异常,就不在继续开启线程。

三、临时变量

我们先来看看下面一段代码:

for (int i = 0; i < 20; i++){    // 开启线程    Task.Run(() =>    {        Task.Run(() => Console.WriteLine($"this is {i}  ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}"));    });}

这段代码的输出结果是什么呢?我们运行程序查看结果:

C#多线程相关操作的示例分析

可能有人会感到疑惑:为什么输出的都是20呢,而不是每次循环变量的值?这是什么原因呢。这是因为我们申请线程的时候不会发生阻塞,而且还是延迟执行的。我们知道,代码的执行速度是非常快的,循环20次几乎一瞬间就完成了,这是i就变成了20,但是线程是延迟执行的,当线程真正去执行的时候,对应的是同一个i,这时i是20,所以输出的都是20。那么该如何输出每次循环的值呢?看下面的代码:

for (int i = 0; i < 20; i++){    // 定义一个新的变量    int k = i;    // 开启线程    Task.Run(() =>    {        Task.Run(() => Console.WriteLine($"this is {i}_{k}  ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}"));    });}

程序运行结果:

C#多线程相关操作的示例分析

这样每次循环的时候,都重新定义变量k,保证每次都是全新的,所以k的值就是每次循环的值。

四、线程安全

什么是线程安全呢?线程安全:如果你的代码在进程中有多个线程同时运行这一段,如果每次运行的结果都跟单线程运行时的结果一致,那么就是线程安全的。

在什么情况下会出现线程安全的问题呢?

一般都是有全局变量/共享变量/静态变量/硬盘文件/数据库的值,只要多线程访问和修改,就会出现线程安全的问题。看下面的代码:

int syncNum = 0;int AsyncNum = 0;for (int i = 0; i < 10000; i++){    syncNum++;}Console.WriteLine($"syncNum={syncNum}"); //单线程10000   10000for (int i = 0; i < 10000; i++){    Task.Run(() =>    {        AsyncNum++;    });}Console.WriteLine($"AsyncNum ={AsyncNum}");

程序运行结果:

C#多线程相关操作的示例分析

这就是线程安全造成的问题。那么该如何解决这个问题呢?这时可以使用lock关键字解决。lock关键字定义如下:

private static readonly object Form_Lock = new object();//锁对象的标准写法

修改代码如下:

int syncNum = 0;int AsyncNum = 0;for (int i = 0; i < 10000; i++){    syncNum++;}Console.WriteLine($"syncNum={syncNum}");for (int i = 0; i < 10000; i++){    Task.Run(() =>    {        lock (Form_Lock)        {            AsyncNum++;        }    });}// 休眠5秒,等待所有线程都执行完毕Thread.Sleep(5000);Console.WriteLine($"AsyncNum ={AsyncNum}");

程序运行结果:

C#多线程相关操作的示例分析

除了使用lock,我们还可以使用数据分拆,避免多线程操作同一个数据,这样又安全又高效。

感谢你能够认真阅读完这篇文章,希望小编分享的“C#多线程相关操作的示例分析”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网行业资讯频道,更多相关知识等着你来学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C#多线程相关操作的示例分析

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C#多线程相关操作的示例分析

这篇文章主要介绍了C#多线程相关操作的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、线程异常我们在单线程中,捕获异常可以使用try-catch,代码如下所示:us
2023-06-29

C#多线程中线程同步的示例分析

这篇文章将为大家详细讲解有关C#多线程中线程同步的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。一、前言我们先来看下面一个例子:using System;using System.Threadin
2023-06-29

CentOS虚拟机下相关操作的示例分析

小编给大家分享一下CentOS虚拟机下相关操作的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!在VM虚拟机中安装完Redhat系统之后如果需要用secureCRT连接linux系统的话操作步骤如下:1.进入linu
2023-06-10

C#中异步多线程的示例分析

这篇文章主要介绍C#中异步多线程的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!进程、线程1. 进程首先了解,什么是线程? 即一个应用程序运行时,占用资源的综合是一个进程。Windows 任务管理器里面可以看
2023-06-25

C++操作符的示例分析

这篇文章给大家介绍C++操作符的示例分析,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。C++操作符是C++中的一个高级C++技术,下面进行相关介绍,或许你发现,有些操作符使用了相同的符号,符号的意义可以随环境进行改变,
2023-06-17

C++常见的stl容器与相关操作示例解析

所谓容器,就是可以承载,包含元素的一个器件,它是STL六大组件之一,是容器、算法、迭代器中最重要也是最核心的一部分
2022-11-13

CentOS7中Xfs文件系统相关操作的示例分析

小编给大家分享一下CentOS7中Xfs文件系统相关操作的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!Xfs是CentOS7.0的默认文件系统类型,而不同文件系统类型对应的创建、检查、调整命令不同,要注意区分。调
2023-06-10

纯C++二叉树相关操作实例代码分析

这篇“纯C++二叉树相关操作实例代码分析”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“纯C++二叉树相关操作实例代码分析”文
2023-07-02

C#异步多线程中Thread的示例分析

这篇文章给大家分享的是有关C#异步多线程中Thread的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。Thread API这里对 Thread 的一些常用 API 进行介绍,使用一些案例进行说明。由于 T
2023-06-25

Python多线程的示例分析

这篇文章将为大家详细讲解有关Python多线程的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。线程相对进程来说是“轻量级”的,操作系统用较少的资源创建和管理线程。程序中的线程在相同的内存空间中执行
2023-06-17

Java多线程的示例分析

这篇文章给大家分享的是有关Java多线程的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。线程的创建1.继承Thread2.实现Runnable3.实现Callable使用继承Thread类来开发多线程的应
2023-06-20

C#多线程举例分析

这篇文章主要讲解了“C#多线程举例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C#多线程举例分析”吧!线程(英语:thread)是操作系统能够进行运算调度的最小单位。它被包含在进程之中
2023-06-22

Python多线程即相关理念分析

这篇文章主要介绍“Python多线程即相关理念分析”,在日常操作中,相信很多人在Python多线程即相关理念分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python多线程即相关理念分析”的疑惑有所帮助!
2023-06-21

iOS中多线程的示例分析

这篇文章给大家分享的是有关iOS中多线程的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、iOS的三种多线程技术1、NSThread–优点:NSThread 比其他两个轻量级,使用简单–缺点:需要自己管
2023-06-21

java中多线程的示例分析

这篇文章主要介绍了java中多线程的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。java多线程并发与并行:并行: 指两个或多个事件在同一时刻发生 ( 同时发生 )
2023-06-20

Java多线程程序的示例分析

今天就跟大家聊聊有关Java多线程程序的示例分析,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。我们在使用Java多线程程序的时候会遇到不少的问题,当我们解决这个问题的时候在源代码中就
2023-06-17

python中队列基本操作和多线程队列的示例分析

这篇文章给大家分享的是有关python中队列基本操作和多线程队列的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、队列基本操作from queue import Queueq = Queue(5) #
2023-06-29

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录