服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C#并发实战记录之Parallel.ForEach使用

C#并发实战记录之Parallel.ForEach使用

2022-08-01 11:12<渔人> C#

这篇文章主要给大家介绍了关于C#并发实战记录之Parallel.ForEach使用的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用C#具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧

前言:

最近给客户开发一个伙食费计算系统,大概需要计算2000个人的伙食。需求是按照员工的预定报餐计划对消费记录进行检查,如有未报餐有刷卡或者有报餐没刷卡的要进行一定的金额扣减等一系列规则。一开始我的想法比较简单,直接用一个for循环搞定,统计结果倒是没问题,但是计算出来太慢了需要7,8分钟。这样系统服务是报超时错误的,让人觉得有点不太爽。由于时间也不多就就先提交给用户使用了,后面逻辑又增加了,计算时间变长,整个计算一遍居然要将近10分钟了。这个对用户来说是能接收的(原来自己手算需要好几天呢),但是我自己接受不了,于是就开始优化了,怎么优化呢,用多线程呗。

一提到多线程,最先想到的是task了,毕竟.net4.0以上task封装了很多好用的方法。但是task毕竟是多开一些线程去执行任务,最后整合结果,这样可以快一些,但我想更加快速一些,于是想到了另外一个对象:parallel。之前在维护代码是确实有遇到过别人写的parallel.invoke,只是指定这个函数的作用是并发执行多项任务,如果遇到多个耗时的操作,他们之间又不贡献变量这个方法不错。我的情况是要并发执行一个集合,于是就用了list.forall 这个方法其实是拓展方法,完整的调用为:list.asparallel().forall,需要先转换成支持并发的集合,等同于parallel.foreach,目的是对集合里面的元素并发执行一系列操作。

于是乎,把原来的foreach换成了list.asparallel().forall,运行起来,果然速度惊人,不到两分钟就插入结果了,但最后却是报主键重复的错误,这个错误的原因是,由于使用了并发,这个时候变量自增,其实是在强着自增,当多个线程同时获取到了id值,都去自增然后就重复了,举个例子如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
int num = 1;
      list<int> list = new list<int>();
      for (int i = 1; i <= 2000; i++)
      {
        list.add(i);
      }
      console.writeline($"num初始值为:" + num.tostring());
      list.asparallel().forall(n =>
      {
        num++;
      });
      console.writeline($"不加锁,并发{list.count}次后为:" + num.tostring());
      console.readkey();

这段代码是让一个变量执行2000次自增,正常结果应该是2001,但实际结果如下:

C#并发实战记录之Parallel.ForEach使用

有经验的同学,立马能想到需要加锁了,c#内置了很多锁对象,如lock 互斥锁,interlocked 内部锁,monitor 这几个比较常见,lock内部实现其实就是使用了monitor对象。对变量自增,interlocked对象提供了,变量自增,自减、或者相加等方法,我们使用自增方法interlocked.increment,函数定义为:int increment(ref int num),该对象提供原子性的变量自增操作,传入目标数值,返回或者ref num都是自增后的结果。 在之前的基础上我们增加一些代码:

?
1
2
3
4
5
6
7
8
num = 1;
      console.writeline($"num初始值为:" + num.tostring());
      list.asparallel().forall(n =>
      {
        interlocked.increment(ref num);
      });
      console.writeline($"使用内部锁,并发{list.count}次后为:" + num.tostring());
      console.readkey();

我们来看运行结果:

C#并发实战记录之Parallel.ForEach使用

加了锁之后id重复算是解决了,其实别高兴太早,由于正常的环境有了id我们还有用这些id来构建对象呢,于是又写了写代码,用集合来添加这些id,为了更真实的模拟生产环境,我在forall里面又加了一层循环代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
num = 1;
      random random = new random();
      var total = 0;
      var m = new concurrentbag<int>();
      list.asparallel().forall(n =>
      {
        var c = random.next(1, 50);
        interlocked.add(ref total, c);
        for (int i = 0; i < c; i++)
        {
          interlocked.increment(ref num);
          m.add(num);
        }
      });
      console.writeline($"使用内部锁,并发+内部循环{list.count}次后为:" + num.tostring());
      console.writeline($"实际值为:{total + 1}");
      var l = m.groupby(n => n).where(o => o.count() > 1);
      console.writeline($"并发里面使用安全集合concurrentbag添加num,集合重复值:{l.count()}个");
      console.readkey();

C#并发实战记录之Parallel.ForEach使用

上面的代码里面我用到了线程安全集合concurrentbag<t>它的命名空间是:using system.collections.concurrent,尽管使用了线程安全集合,但是在并发面前仍然是不安全的,到了这里其实比较郁闷了,自增加锁,安全集合内部应该也使用了锁,但还是重复了。有点说不过去了,想想多线程执行时有个上下文对象,即当多个线程同时执行任务,共享了变量他们一开始传进去的对象数值应该是相同的,由于变量自增时加了锁,所以id是不会重复了。我猜测问题应该出在add方法了,就是说当num值自增后还没有来得及传出去就已经执行了add方法,故添加了重复变量。于是乎,我重新写了段代码,让id自增和集合添加都放到锁里面:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
num = 1;
      total = 0;
      using (var q = new blockingcollection<int>())
      {
        list.asparallel().forall(n =>
        {
          var c = random.next(1, 50);
          interlocked.add(ref total, c);
          for (int i = 0; i < c; i++)
          {
            
            // task.delay(100);
            q.add(interlocked.increment(ref num));
            
            //可控
            //lock (objlock)
            //{
            //  num++;
            //  q.add(num);
            //}
          }
 
        });
        q.completeadding();
        console.writeline($"num累计值为:{total},并发之后值为:{num}");
        var x = q.groupby(n => n).where(o => o.count() > 1);
        console.writeline($"并发使用安全集合blockingcollection+interlocked添加num,集合重复值:{x.count()}个");
        console.readkey();
      }

这里我测试了另外一个线程安全的集合blockingcollection,关于这个集合的使用请自行查找msdn文档,上面的关键代码直接添加安全集合的返回值,可以保证集合不会重复,但其实下面的lock更适用与正式环境,因为我们添加的一般都是对象不会是基础类型数值,运行结果如下:

C#并发实战记录之Parallel.ForEach使用

至此,我们的问题解决了,计算时间由原来的9分多降至110秒左右,可见parallel的处理还是很给力的,唯一不足的是,很占cpu,执行计算后cpu达到了88%。附上计算结果:

C#并发实战记录之Parallel.ForEach使用

优化前后对比

C#并发实战记录之Parallel.ForEach使用

总结:

c#安全集合在并发的情况下其实不一定是安全的,还是需要结合实际应用场景和验证结果为准。parallel.foreach在对循环数量可观的情况下是可以去使用的,如果有共享变量,一定要配合锁做同步处理。还是得慎用这个方法,如果方法内部有操作数据库的记得增加事务处理,否则就呵呵了。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对服务器之家的支持。

原文链接:https://www.cnblogs.com/heweijian/p/11330282.html

延伸 · 阅读

精彩推荐
  • C#通过C#实现发送自定义的html格式邮件

    通过C#实现发送自定义的html格式邮件

    本篇文章主要介绍了通过C#实现发送自定义的html格式邮件,详细的介绍了发送HTML格式邮件的方法,有兴趣的可以了解一下。...

    柔城5862021-12-22
  • C#C#使用dynamic类型访问JObject对象

    C#使用dynamic类型访问JObject对象

    这篇文章主要为大家详细介绍了C#使用dynamic类型访问JObject对象,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    ZKEASOFT10122022-02-22
  • C#C#版Windows服务安装卸载小工具

    C#版Windows服务安装卸载小工具

    这篇文章主要为大家推荐了一款C#版Windows服务安装卸载小工具,小巧灵活的控制台程序,希望大家喜欢,感兴趣的小伙伴们可以参考一下...

    韩天伟7572021-12-01
  • C#C#使用LitJson解析JSON的示例代码

    C#使用LitJson解析JSON的示例代码

    本篇文章主要介绍了C#使用LitJson解析JSON的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    peiandsky6282021-12-22
  • C#C#集合遍历时删除和增加元素的方法

    C#集合遍历时删除和增加元素的方法

    这篇文章主要介绍了C#集合遍历时删除和增加元素的方法,结合实例形式分析了C#针对集合元素的遍历、添加与删除等操作实现方法与注意事项,需要的朋友可...

    codingsilence8472021-11-29
  • C#C#使用Aforge调用摄像头拍照的方法

    C#使用Aforge调用摄像头拍照的方法

    这篇文章主要为大家详细介绍了C#使用Aforge调用摄像头拍照的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    Jichan·Jong3712022-03-01
  • C#c#爬虫爬取京东的商品信息

    c#爬虫爬取京东的商品信息

    这篇文章主要给大家介绍了关于利用c#爬虫爬取京东商品信息的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习...

    alone_alone3482022-03-05
  • C#C#获取网页源代码的方法

    C#获取网页源代码的方法

    这篇文章主要介绍了C#获取网页源代码的方法,涉及C#基于自定义函数读取网页html代码的方法,具有一定参考借鉴价值,需要的朋友可以参考下...

    C#教程网9582021-10-26