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();
        }
    }
}

程序运行结果:

那么在多线程中如何捕获异常呢?是不是也可以使用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();
        }
    }
}

程序运行结果:

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

我们看到程序中确实也抛出了异常,但是程序却没有捕获到,那么异常去哪里了呢?异常被多线程给吞掉了,那么如何在多线程中捕获异常呢?如果把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}");
}

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

这时就可以进入到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}");
}

最后运行程序:

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

二、线程取消

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

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}");
            }
     });
}

程序运行结果:

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

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

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

所以,我们可以利用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);
    }
}

程序运行结果:

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

三、临时变量

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

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

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

可能有人会感到疑惑:为什么输出的都是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")}"));
    });
}

程序运行结果:

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

四、线程安全

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

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

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

int syncNum = 0;

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

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

程序运行结果:

这就是线程安全造成的问题。那么该如何解决这个问题呢?这时可以使用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}");

程序运行结果:

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

到此这篇关于C#多线程相关操作的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 深入学习C#多线程

    目录 一.基本概念 1.进程 2.线程 二.多线程 2.1System.Threading.Thread类 2.2 线程的常用属性 2.2.1线程的标识符 2.2.2线程的优先级别 2.2.3线程的状态 2.2.4System.Threading.Thread的方法 2.3前台线程和后台线程 2.4线程同步 2.5跨线程访问 2.6终止线程 三.同步和异步 四.回调 获取委托异步调用的返回值 一.基本概念 1.进程 首先打开任务管理器,查看当前运行的进程: 从任务管理器里面可以看到当前所有正在运

  • C#多线程系列之async和await用法详解

    目录 async和await async await 从以往知识推导 创建异步任务 创建异步任务并返回Task 异步改同步 说说awaitTask 说说 asyncTask<TResult> 同步异步? Task封装异步任务 关于跳到await变异步 为什么出现一层层的await async和await async 微软文档:使用 async 修饰符可将方法.lambda 表达式或匿名方法指定为异步. 使用 async 修饰的方法,称为异步方法. 例如: 为了命名规范,使用 async 修饰的

  • C#多线程系列之线程完成数

    解决一个问题 假如,程序需要向一个 Web 发送 5 次请求,受网路波动影响,有一定几率请求失败.如果失败了,就需要重试. 示例代码如下: class Program { private static int count = 0; static void Main(string[] args) { for (int i = 0; i < 5; i++) new Thread(HttpRequest).Start(); // 创建线程 // 用于不断向另一个线程发送信号 while (count

  • C#多线程系列之线程通知

    AutoRestEvent 类用于从一个线程向另一个线程发送通知. 微软文档是这样介绍的:表示线程同步事件在一个等待线程释放后收到信号时自动重置. 其构造函数只有一个: 构造函数里面的参数用于设置信号状态. 构造函数 说明 AutoResetEvent(Boolean) 用一个指示是否将初始状态设置为终止的布尔值初始化 AutoResetEvent 类的新实例. 真糟糕的机器翻译. 常用方法 AutoRestEvent 类是干嘛的,构造函数的参数又是干嘛的?不着急,我们来先来看看这个类常用的方法

  • C#多线程系列之工作流实现

    目录 前言 节点 Then Parallel Schedule Delay 试用一下 顺序节点 并行任务 编写工作流 接口构建器 工作流构建器 依赖注入 实现工作流解析 前言 前面学习了很多多线程和任务的基础知识,这里要来实践一下啦.通过本篇教程,你可以写出一个简单的工作流引擎. 本篇教程内容完成是基于任务的,只需要看过笔者的三篇关于异步的文章,掌握 C# 基础,即可轻松完成. C#多线程系列之任务基础(一) C#多线程系列之任务基础(二) C#多线程系列之任务基础(三) 由于本篇文章编写的工作

  • C#多线程编程Task用法详解

    目录 一.基本概念 Task优势 二.Task用法 创建任务 1.使用Task创建无返回值 2.使用Task.Run方法创建任务 3.使用Factory方式创建任务 4.创建带返回值的Task 三.常见方法 1.WaitAll() 2.WaitAny() 3.ContinueWhenAll() 4.ContinueWhenAny 5.ContinueWith 一.基本概念 Task优势 ThreadPool相比Thread来说具备了很多优势,但是ThreadPool却又存在一些使用上的不方便,例

  • C#多线程系列之线程等待

    目录 前言 volatile 关键字 三种常用等待 再说自旋和阻塞 SpinWait 结构 属性和方法 自旋示例 新的实现 SpinLock 结构 属性和方法 示例 等待性能对比 前言 volatile 关键字 volatile 关键字指示一个字段可以由多个同时执行的线程修改. 我们继续使用<C#多线程(3):原子操作>中的示例: static void Main(string[] args) { for (int i = 0; i < 5; i++) { new Thread(AddO

  • C#多线程之线程池ThreadPool用法

    目录 一.ThreadPool 1.QueueUserWorkItem() 2.GetMaxThreads() 3.GetMinThreads() 4.SetMaxThreads()和SetMinThreads() 二.线程等待 三.线程重用 一.ThreadPool ThreadPool是.Net Framework 2.0版本中出现的. ThreadPool出现的背景:Thread功能繁多,而且对线程数量没有管控,对于线程的开辟和销毁要消耗大量的资源.每次new一个THread都要重新开辟内

  • C#多线程系列之读写锁

    本篇的内容主要是介绍 ReaderWriterLockSlim 类,来实现多线程下的读写分离. ReaderWriterLockSlim ReaderWriterLock 类:定义支持单个写线程和多个读线程的锁. ReaderWriterLockSlim 类:表示用于管理资源访问的锁定状态,可实现多线程读取或进行独占式写入访问. 两者的 API 十分接近,而且 ReaderWriterLockSlim 相对 ReaderWriterLock 来说 更加安全.因此本文主要讲解 ReaderWrit

  • C#多线程系列之多阶段并行线程

    前言 这一篇,我们将学习用于实现并行任务.使得多个线程有序同步完成多个阶段的任务. 应用场景主要是控制 N 个线程(可随时增加或减少执行的线程),使得多线程在能够在 M 个阶段中保持同步. 线程工作情况如下: 我们接下来 将学习C# 中的 Barrier ,用于实现并行协同工作. Barrier 类 使多个任务能够采用并行方式依据某种算法在多个阶段中协同工作,使多个线程(称为“参与者” )分阶段同时处理算法. 可以使多个线程(称为“参与者” )分阶段同时处理算法.(注意算法这个词) 每个参与者完

  • C#多线程系列之线程池

    目录 线程池 ThreadPool 常用属性和方法 线程池说明和示例 线程池线程数 线程池线程数说明 不支持的线程池异步委托 任务取消功能 计时器 线程池 线程池全称为托管线程池,线程池受 .NET 通用语言运行时(CLR)管理,线程的生命周期由 CLR 处理,因此我们可以专注于实现任务,而不需要理会线程管理. 线程池的应用场景:任务并行库 (TPL)操作.异步 I/O 完成.计时器回调.注册的等待操作.使用委托的异步方法调用和套接字连接. 很多人不清楚 Task.Task<TResult>

  • C#多线程系列之手动线程通知

    区别与示例 AutoResetEvent 和 ManualResetEvent 十分相似.两者之间的区别,在于前者是自动(Auto),后者是手动(Manua). 你可以先运行下面的示例,再测试两者的区别. AutoResetEvent 示例: class Program { // 线程通知 private static AutoResetEvent resetEvent = new AutoResetEvent(false); static void Main(string[] args) {

随机推荐