使用多线程处理List数据

最近遇到了一个业务场景,需要对List中的数据逐个发起http请求(List中的数据各自独立,对执行顺序无要求),考虑到可以使用多线程加快处理速度。

封装了如下方法:

cs 复制代码
        /// <summary>
        /// 多线程处理数据-无返回值
        /// </summary>
        /// <param name="list"></param>
        /// <param name="action">数据处理方法</param>
        /// <param name="threadCount">开启线程数量</param>
        /// <param name="waitFlag">是否等待全部结束</param>
        static void RunTask<T>(List<T> list, Action<T> action, int threadCount = 5, bool waitFlag = true)
        {
            var queue = new ConcurrentQueue<T>(list);
            threadCount = Math.Min(threadCount, list.Count);
            Task[] tasks = new Task[threadCount];
            for (int i = 0; i < threadCount; i++)
            {
                tasks[i] = Task.Run(() =>
                {
                    while (queue.TryDequeue(out T t))
                    {
                        action(t);
                    }
                });
            }
            if (waitFlag)
            {
                Task.WaitAll(tasks);
            }
        }

        /// <summary>
        /// 多线程处理数据-有返回值
        /// </summary>
        /// <param name="list">待处理数据</param>
        /// <param name="func">数据处理方法</param>
        /// <param name="count">开启线程数量</param>
        static List<TReturn> RunTask<T, TReturn>(List<T> list, Func<T, TReturn> func, int threadCount = 5)
        {
            var result = new ConcurrentBag<TReturn>();
            var queue = new ConcurrentQueue<T>(list);
            threadCount = Math.Min(threadCount, list.Count);
            var tasks = new Task[threadCount];
            for (int i = 0; i < threadCount; i++)
            {
                tasks[i] = Task.Run(() =>
                {
                    while (queue.TryDequeue(out T t))
                    {
                        result.Add(func(t));
                    }
                });
            }
            Task.WaitAll(tasks);
            return result.ToList();
        }

使用示例:

cs 复制代码
        public void DoSingle(User user, string param1, string param2)
        {
            //HttpHelper.Get("http://xxxx/userid="+user.id);
            //doSomething();
            Console.WriteLine(user.Id + "-" + user.Name + "-" + param1 + "-" + param2);
        }
        public string DoSingle2(User user, string param1, string param2)
        {
            //HttpHelper.Get("http://xxxx/userid="+user.id);
            //doSomething();
            Console.WriteLine(user.Id + "-" + user.Name + "-" + param1 + "-" + param2);
            return user.Id + "-666";
        }


        public class User
        {
            /// <summary>
            /// id
            /// </summary>
            public int Id { get; set; }
            /// <summary>
            /// 姓名
            /// </summary>
            public string Name { get; set; }
        }

        public void Test()
        {
            var list = new List<User> {
                new User { Id = 1, Name = "张三" },
                new User { Id = 2, Name = "李四" },
                new User { Id = 3, Name = "王五" }
            };
            RunTask(list, t => DoSingle(t, "参数1", "参数2"));
            var retList = RunTask(list, t => DoSingle2(t, "参数1", "参数2"));
        }
相关推荐
SunnyDays10112 小时前
如何使用 C# 将 CSV 数据轻松转换为 PDF
c#·csv转pdf
Big_潘大师2 小时前
C# 六自由度机械臂正反解计算
数学建模·机器人·c#·六自由度机械臂
我不是程序猿儿2 小时前
【C#】XtraMessageBox(DevExpress)与MessageBox(WinForms 标准库)的区别
开发语言·c#
偶尔的鼠标人2 小时前
Avalonia/WPF 打开子窗口,并且跨页面传值
c#·wpf·mvvm·avalonia
玖笙&2 小时前
✨WPF编程进阶【6.1】:图形原则(附源码)
c++·c#·wpf·visual studio
huoshan123453 小时前
给旧版 .NET 也开一扇“私有之门”——ILAccess.Fody 实现原理与设计
c#·.net·fody·il·mono.cecil
纸照片3 小时前
WPF中为Button设置IsMouseOver和IsPressed事件中改变背景颜色不起作用
c#·.net·wpf
关关长语4 小时前
Dotnet使用System.Xml.Serialization处理Xml序列化
xml·c#·.net
歪歪1007 小时前
在C#中除了按属性排序,集合可视化器还有哪些辅助筛选的方法?
开发语言·前端·ide·c#·visual studio
weixin_307779137 小时前
C#程序实现将Teradata的存储过程转换为Snowflake的sql的存储过程
数据库·数据仓库·c#·云计算·迁移学习