看了第一篇文章,多线程系列,看到了在线程执行任务队列有了一定的了解~!

那么今天我来讲讲,怎么样构建通用的自定义线程概念!

线程执行任务,肯定要有目标,但是如果写死了,那么一个线程处理执行职能按照思路处理一类任务,显然不满足我们的实际场景的需求,那么怎么才能创建灵活的线程执行器呢!

首先我们来创建一个任务构造器!

   /// <summary>
     /// 线程模型执行任务 基类
     /// </summary>
     public abstract class BaseTask
     {

         /// <summary>
         /// 任务ID
         /// </summary>
         public long TID { get; set; }

         /// <summary>
         /// 任务名称
         /// </summary>
         public string TName { get; set; }

         /// <summary>
         /// 线程模型任务
         /// </summary>
         public abstract void Run();

         public override string ToString()
         {
             return "Task<" + this.TName + "(" + TID + ")>";
         }
     }

看到这里,可能不会明白,这么写法的意义在哪里呢?

那么我们再来自定义线程的执行器

 /// <summary>
     /// 定义自定义线程模型
     /// </summary>
     public abstract class BaseThread<T> where T : BaseTask
     {
         //执行线程
         Thread _Thread;
         //通知一个或多个正在等待的线程已发生事件
         ManualResetEvent mre = new ManualResetEvent(false);
         //线程安全的队列
         System.Collections.Concurrent.ConcurrentQueue<T> cqueue = new System.Collections.Concurrent.ConcurrentQueue<T>();

         /// <summary>
         /// 自定义线程ID;
         /// </summary>
         public long TID { get; set; }

         public static bool IsRuning = true;

         /// <summary>
         /// 初始化
         /// </summary>
         /// <param name="tName">线程的名称</param>
         public BaseThread(string tName)
         {
             _Thread = new Thread(Runing);
             _Thread.Name = tName;
             _Thread.Start();
         }

         //模拟新增任务
         public void AddTask(T task)
         {
             //添加任务到队列
             cqueue.Enqueue(task);
             //唤醒所有相关的挂起线程
             mre.Set();
         }

         void Runing()
         {
             //主循环 服务器运行标识
             while (IsRuning)
             {
                 //如果是空则继续等待      服务器运行标识
                 while (cqueue.IsEmpty && IsRuning)
                 {
                     //重置线程暂停状态
                     mre.Reset();
                     //这个操作是以便服务器需要停止操作,
                     //如果停止调用线程的Thread.Abort()是会导致处理队列任务丢失
                     mre.WaitOne();
 #if DEBUG
                     //为了方便测试信息打印的暂停信息
                     Console.WriteLine(DateTime.Now.ToString("HH:mm:ss:ffff") + " : " + Thread.CurrentThread.Name + " Status Sleep");
 #endif
                 }
                 T t;
                 //取出队列任务
                 if (cqueue.TryDequeue(out t))
                 {
                     Runing(t);
                 }
             }
         }

         /// <summary>
         /// 设置运行方法为虚方法,方便子函数覆盖
         /// </summary>
         protected virtual void Runing(T run)
         {
             try
             {
                 //执行任务
                 run.Run();
                 //打印任务信息
                 Console.WriteLine(DateTime.Now.ToString("HH:mm:ss:ffff") + " : " + Thread.CurrentThread.Name + " Action:" + run.ToString());
             }
             catch (Exception ex)
             {
                 //打印任务信息
                 Console.WriteLine(DateTime.Now.ToString("HH:mm:ss:ffff") + " : " + Thread.CurrentThread.Name + " Action:" + run.ToString() + " Exception:" + ex);
             }
         }
     }

看到这里是不是比较清楚了?这样我们定义的线程是不是能完成多不同类型的任务呢?

是不是可以做到一个线程执行和处理不同类型的任务呢?

接下来我们创建类型的实现类

任务实现类1

 /// <summary>
     /// 测试任务
     /// </summary>
     public class TestTask : BaseTask
     {

         public override void Run()
         {
             Console.WriteLine("我只是用来测试的");
         }
     }

任务实现类2

 public class TestTask1 : BaseTask
     {

         /// <summary>
         /// 执行任务
         /// </summary>
         public Action Test { get; set; }

         public override void Run()
         {
             if (Test != null)
             {
                 Test();
             }
             Console.WriteLine("我只是用来测试的");
         }
     }

线程的实现类

     /// <summary>
     /// 测试线程
     /// </summary>
     public class TestThread : BaseThread<BaseTask>
     {
         public TestThread()
             : base("TestThread")
         {

         }
     }

接下来我们看看测试效果

     class Program
     {

         static void Main(string[] args)
         {
             TestThread tt = new TestThread();
             tt.AddTask(, TName = "测试1" });
             tt.AddTask(, TName = "测试2" });
             tt.AddTask(, TName = ); }) });
             tt.AddTask(, TName = "测试4" });
             Console.ReadLine();
         }
     }

运行结果图

看到这里是不是很清楚明了呢?这样我们在处理日常任务的时候,不同类型的逻辑或者任务类型,是不是都可以放到一个线程执行呢?

请大家期待我的下一篇文章,为大家讲解,定时器线程执行器,

c# 多线程系列二 自定义线程执行器的更多相关文章

  1. (Java多线程系列二)线程间同步

    Java多线程间同步 1.什么是线程安全 通过一个案例了解线程安全 案例:需求现在有100张火车票,有两个窗口同时抢火车票,请使用多线程模拟抢票效果. 先来看一个线程不安全的例子 class Sell ...

  2. java多线程系列(二)

    对象变量的并发访问 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我 ...

  3. java多线程系列(二)---对象变量并发访问

    对象变量的并发访问 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我 ...

  4. java多线程编程(二创建线程)

    1.概念           因为java是完全面向对象的,所以在java中,我们说的线程,就是Thread类的一个实例对象.所以,一个线程就是一个对象,它有自己字段和方法. 2.创建线程 创建线程有 ...

  5. Java多线程(二) —— 线程安全、线程同步、线程间通信(含面试题集)

    一.线程安全 多个线程在执行同一段代码的时候,每次的执行结果和单线程执行的结果都是一样的,不存在执行结果的二义性,就可以称作是线程安全的. 讲到线程安全问题,其实是指多线程环境下对共享资源的访问可能会 ...

  6. java多线程(四)-自定义线程池

    当我们使用 线程池的时候,可以使用 newCachedThreadPool()或者 newFixedThreadPool(int)等方法,其实我们深入到这些方法里面,就可以看到它们的是实现方式是这样的 ...

  7. java多线程(二)-线程的生命周期及线程间通信

    一.摘要    当我们将线程创建并start时候,它不会一直占据着cpu执行,而是多个线程间会去执行着这个cpu,此时这些线程就会在多个状态之间进行着切换. 在线程的生命周期中,它会有5种状态,分别为 ...

  8. java多线程系列六、线程池

    一. 线程池简介 1. 线程池的概念: 线程池就是首先创建一些线程,它们的集合称为线程池. 2. 使用线程池的好处 a) 降低资源的消耗.使用线程池不用频繁的创建线程和销毁线程 b) 提高响应速度,任 ...

  9. Java 多线程(二)—— 线程的同步

     上文创建多线程买票的例子中注释会出现错票.重票的问题,本文来讲讲如何解决此问题.本文例子:利用多线程模拟 3 个窗口卖票 实现Runnable接口 public class TestThread2 ...

随机推荐

  1. 普通用户使用dbms_xplan包需要有的权限

    普通用户使用dbms_xplan包查看执行计划需要对v$sql.v$sql_plan.v$session及v$sql_plan_statistics_all这四个视图同时具有select权限. 如果普 ...

  2. AWIT DBackup 0.0.20 发布,备份系统

    AWIT DBackup 0.0.20 修复了几个小 bug. AllWorldIT DBackup 是一个备份系统,为每个目录创建一个独立的压缩包,这更便于搜索. 特点: 使用 xz, bzip2, ...

  3. 作业三:PSP记录耗时情况

    PSP2.1 Personal Software Process Stage Time planning 计划 15min Estimate 估计这个任务多久完成 130min Developing ...

  4. [SQLServer大对象]——FileTable初体验

    阅读导航 启用FILESTREAM设置 更改FILESTRAM设置 启用数据库非事务性访问级别 FileTable 在我接触FileTable之前,存储文件都是存储文件的链接和扩展名到数据,其实并没有 ...

  5. NanoProfiler - 适合生产环境的性能监控类库 之 基本功能篇

    背景 NanoProfiler是一个EF Learning Labs出品的免费性能监控类库(即将开源).它的思想和使用方式类似于MiniProfiler的.但是,设计理念有较大差异. MiniProf ...

  6. 网络误区:不用中间变量交换2个变量的value,最高效的是异或运算.

    本文记录了不使用中间变量交换2个变量的value,很多的网络留言说是直接异或运算就可以了,而且效率很高,是真的吗? 这里简单的说一下我的环境:Win7 32位,Qt creator 5.4.1 编译器 ...

  7. EF架构~DefaultValue让我的UnitOfWork更可读

    回到目录 在编程世界里,使用“否定式”不是一件好事,因为它的可读性并不高,但有时,为了让使用者减少代码量,还是使用了双重否定,直到DefaultValue的出现,它改变了这一切,它可以为你的属性设置你 ...

  8. JAVA数据类型,变量,转换,常量,运算符

    java数据类型: Java基本类型共有八种,基本类型可以分为三类: 1.字符类型char,用单引号赋值 2.布尔类型boolean 3.数值类型byte.short.int.long.float.d ...

  9. paip. 解决php 以及 python 连接access无效的参数量。参数不足,期待是 1”的错误

    paip. 解决php 以及 python 连接access无效的参数量.参数不足,期待是 1"的错误 作者Attilax  艾龙,  EMAIL:1466519819@qq.com  来源 ...

  10. Eclipse中java向数据库中添加数据,更新数据,删除数据

    前面详细写过如何连接数据库的具体操作,下面介绍向数据库中添加数据. 注意事项:如果参考下面代码,需要 改包名,数据库名,数据库账号,密码,和数据表(数据表里面的信息) package com.ning ...