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

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

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

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

   /// <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. 酷狗 KRC 文件的解析

    清理硬盘发现以前写过一个进行一半的代码,这次补全并从硬盘删掉. 格式说明来自 https://shansing.com/read/392/ krc解码并解压缩后得到一个字符串,例子: [id:$000 ...

  2. Java设计模式12:装饰器模式

    装饰器模式 装饰器模式又称为包装(Wrapper)模式.装饰器模式以多客户端透明的方式扩展对象的功能,是继承关系的一个替代方案. 装饰器模式的结构 通常给对象添加功能,要么直接修改对象添加相应的功能, ...

  3. Winform启动隐藏,WebBrowser交互JS

    一.启动隐藏 Winform比较奇怪,Load的时候设置Visiable=false,无效.webBrowser_DocumentCompleted之后调用hide隐藏了窗体,但是在notifyIco ...

  4. TW2015技术雷达中文版发布

    今天thoughtworks 2015新版技术雷达pdf发布了,你可以从这里下载http://engage.thoughtworks.com/HQ0000Q0QOf5pE70nbD00GP,在这里你可 ...

  5. java实现输入一行字符,分别统计出其中英文字母、空格、数字和其它字符的个数。

    package JingDian; import java.util.Scanner; public class charKind { public static void main(String[] ...

  6. Fragment提交transaction导致state loss异常

    下面自从Honeycomb发布后,下面栈跟踪信息和异常信息已经困扰了StackOverFlow很久了. java.lang.IllegalStateException: Can not perform ...

  7. MySQL账户安全设置

    一般来说,安装完MySQL后,默认的用户是root,密码123456,外网不能访问. 有时候也希望在外网访问,则可以添加一个账户.建议不要用root. 如下表,存在安全问题: mysql> se ...

  8. Oracle Redo 以及 Archived日志简述

    Oracle通过Redo Archived实现数据的归档 什么是Redo日志 Redo日志记录了数据的变更,用于在数据库出现故障后,进行数据恢复. 功能主要由三个组件实现:Redo Log Buffe ...

  9. Atitit图像识别的常用特征大总结attilax大总结

    Atitit图像识别的常用特征大总结attilax大总结 1.1. 常用的图像特征有颜色特征.纹理特征.形状特征.空间关系特征. 1 1.2. HOG特征:方向梯度直方图(Histogram of O ...

  10. Ext2.x学习笔记

    Ext2.X学习笔记一 一.ExtJS简介  1.1 什么是Ext JS? · Ext JS是一个Ajax框架,可以用来开发富客户端的Ajax应用,是一个用javascript写的,主要用于创建前端用 ...