如果需要查看更多文章,请微信搜索公众号 csharp编程大全,需要进C#交流群群请加微信z438679770,备注进群, 我邀请你进群! ! !

------------------------------------------------------------------------------------------------------------------------------------------------

1.什么是thread
当我们提及多线程的时候会想到thread和threadpool,这都是异步操作,threadpool其实就是thread的集合,具有很多优势,不过在任务多的时候全局队列会存在竞争而消耗资源。thread默认为前台线程,主程序必须等线程跑完才会关,而threadpool相反。
总结:threadpool确实比thread性能优,但是两者都没有很好的api区控制,如果线程执行无响应就只能等待结束,从而诞生了task任务。
2.什么是task
 task简单地看就是任务,那和thread有什么区别呢?Task的背后的实现也是使用了线程池线程,但它的性能优于ThreadPoll,因为它使用的不是线程池的全局队列,而是使用的本地队列,使线程之间的资源竞争减少。同时Task提供了丰富的API来管理线程、控制。但是相对前面的两种耗内存,Task依赖于CPU对于多核的CPU性能远超前两者,单核的CPU三者的性能没什么差别。

Task其实就是在ThreadPool的基础上进行一层封装,ThreaPool启动的线程不好判断线程的执行情况,但Task可以,很好地解决了这个问题。

Thread与ThreadPoll

前台线程:主程序必须等待线程执行完毕后才可退出程序。Thread默认为前台线程,也可以设置为后台线程

后台线程:主程序执行完毕后就退出,不管线程是否执行完毕。ThreadPool默认为后台线程

线程消耗:开启一个新线程,线程不做任何操作,都要消耗1M左右的内存

ThreadPoll是线程池 其目的是为了减少开启新线程消耗的资源(使用线程池中的空闲线程,不必在开启新线程,以及统一管理线程(线程池中的线程执行完毕后,回归到线程池里,等待新任务).

总结:ThreadPoll性能优于Thread,但是Thread和ThreadPoll对线程的控制都不是很好,例如线程等待(线程执行一段时间无响应后,直接停止线程,释放资源 等 都没有直接的API来控制 只能通过硬编码来实现,同时ThreadPool使用的是线程池全局队列,全局队列中的线程依旧会存在竞争共享资源的情况,从而影响性能。

然后task

Task的背后的实现也是使用了线程池线程,但它的性能优于ThreadPoll,因为它使用的不是线程池的全局队列,而是使用的本地队列,使线程之间的资源竞争减少。同时Task提供了丰富的API来管理线程、控制。但是相对前面的两种耗内存,Task依赖于CPU对于多核的CPU性能远超前两者,单核的CPU三者的性能没什么差别。

Task不等于Thread,只是微软默认实现ThreadPoolTaskScheduler是依赖于线程池的,因为该类的可访问性为internal,所以我们在实际编码中无法直接在代码中new这么一个Scheduler出来,只能通过TaskScheduler.Default间接的来使用

构造函数:

public Thread (System.Threading.ThreadStart start);
无参数

public Thread (System.Threading.ParameterizedThreadStart start);
有参数

属性:

IsBackground
获取或设置线程是否为后台线程

Priority
获取或设置优先级

ManagedThreadId
获取当前线程的唯一标识符

方法:

Abort()
终止线程

Join()

让线程依次运行(这个方法经常用到)

使用:

//无参数的线程
Thread thread=new Thread(new ThreadStart(方法名));//实例化线程
thread.Start();//启动线程 //有参数的线程
Thread threadParam = new Thread(new ParameterizedThreadStart(方法名));//有参数
//////这里有个非常重要的知识
方法里面的形参必须是object类型的
threadParam.Start(DateTime.Now);//有参数的线程启动方法

  

ThreadPool

提供一个线程池,该线程池可用于执行任务、发送工作项、处理异步 I/O、代表其他线程等待以及处理计时器。

方法:

QueueUserWorkItem(WaitCallback)
将方法排入队列以便执行

QueueUserWorkItem(WaitCallback, Object)
将方法排入队列以便执行,Object需要传递的参数

ThreadPool.QueueUserWorkItem(方法名);//这个方法必须要有个参数object

Task

表示一个异步操作。

构造函数:

public Task (Action action);
无参数无返回值

public Task(Action action, object state);
有参数无返回值

public Task(Func<object, TResult> function, object state);
有参数有返回值

属性:

CurrentId
正在执行的Task的id

IsCompleted
是否完成

IsCompleted
是否出现异常

方法:

Start()
启动Task

Wait()
等待Task执行完成

使用:

Task task_NoParam = new Task(无参数无返回值的方法);
task_NoParam.Start(); Task task_WithParam = new Task(有参数无返回值的方法, 传给方法的参数);
task_WithParam.Start(); Task<string> task_WithParam_WithReturn = new Task<string>(有参数有返回值的方法, 传给方法的参数);
task_WithParam_WithReturn.Start();
string Result=task_WithParam_WithReturn.Result;//返回的结果

  三、具体代码

class CommonClass
{
public void TestMethod()
{
Console.WriteLine("没有参数的方法");
for (int i = 0; i < 3; i++)
{
Console.WriteLine("无参数的方法" + i + "");
}
} public void TestMethod(object obj)//这个形参必须是object类型的---这很重要
{
Console.WriteLine("有参数的方法,参数为" + obj.ToString() + "");
for (int i = 0; i < 3; i++)
{
Console.WriteLine("有参数的方法" + i + "");
}
}
public void TestMethod_ThreadPool(object obj)
{
if (obj != null)
{
Console.WriteLine("ThreadPool-有参数的方法,参数为" + obj.ToString() + "");
for (int i = 0; i < 3; i++)
{
Console.WriteLine("ThreadPool-有参数的方法" + i + "");
}
}
else
{
Console.WriteLine("ThreadPool-没有参数的方法");
for (int i = 0; i < 3; i++)
{
Console.WriteLine("ThreadPool-无参数的方法" + i + "");
}
} }
public void TestMethod_Task_NoParam()
{
Console.WriteLine("Task-无参数");
}
public void TestMethod_Task_WithParam(object obj)
{
Console.WriteLine($"Task-有参数,参数为:{obj.ToString()}");
}
public string TestMethod_Task_WithParam_WithReturn(object obj)
{
Console.WriteLine($"Task-有参数,参数为:{obj.ToString()}");
return obj.ToString();
}
}

  主程序代码:

class Program
{
static void Main()
{
for (int i = 0; i < 3; i++)
{
Console.WriteLine("主线程" + i + "");
}
CommonClass commonClass = new CommonClass(); //Thread的使用
Thread thread = new Thread(new ThreadStart(commonClass.TestMethod));//没有参数
//thread.IsBackground = false;//设置前台线程还是后台线程,在线程启动前设置
thread.Start();//没有参数的线程启动方法 Thread threadParam = new Thread(new ParameterizedThreadStart(commonClass.TestMethod));//有参数
threadParam.Start(DateTime.Now);//有参数的线程启动方法 //等待上面两个线程执行完后
thread.Join();
threadParam.Join(); Console.WriteLine("\n下面是ThreadPool的使用");
//ThreadPool的使用
ThreadPool.QueueUserWorkItem(commonClass.TestMethod_ThreadPool);
ThreadPool.QueueUserWorkItem(commonClass.TestMethod_ThreadPool, DateTime.Now);
//注意:使用ThreadPool不好判断线程什么时候完成 Thread.Sleep(1000);
Console.WriteLine("\n下面是Task的使用");
//Task的使用
Task task_NoParam = new Task(commonClass.TestMethod_Task_NoParam);//无参数无返回值的方法
task_NoParam.Start();
Task.WaitAll(task_NoParam);//等task_NoParam这个Task执行完执行下面的
//这就是使用Task的好处,便于控制,知道Task什么时候执行完
//不像TreadPool,让他启动后台线程,然后就没有然后了。任务完成后自动销毁。
Task task_WithParam = new Task(commonClass.TestMethod_Task_WithParam, "sdf");//有参数无返回值的方法
task_WithParam.Start();
Task<string> task_WithParam_WithReturn = new Task<string>(commonClass.TestMethod_Task_WithParam_WithReturn, "sfdgsdfgasdf");//有参数有返回值
task_WithParam_WithReturn.Start();
Console.WriteLine("有参数有返回值的Task执行结果:" + task_WithParam_WithReturn.Result + ""); Console.ReadKey();
}
}

  结果:

如果需要查看更多文章,请微信搜索公众号 csharp编程大全,需要进C#交流群群请加微信z438679770,备注进群, 我邀请你进群! ! !

c#之task与thread区别及其使用的更多相关文章

  1. Task与Thread间的区别

    通过查找一些文章,得知,Task与Thread不可比.Task是为了利用多CPU多核的机制而将一个大任务不断分解成小任务,这些任务具体由哪一个线程或当前线程执行由OS来决定.如果你想自己控制由哪一个T ...

  2. Linux中的task,process, thread 简介

    本文的主要目的是介绍在Linux内核中,task,process, thread这3个名字之间的区别和联系.并且和WINDOWS中的相应观念进行比较.如果你已经很清楚了,那么就不用往下看了. LINU ...

  3. C# Task.Run 和 Task.Factory.StartNew 区别

    Task.Run 是在 dotnet framework 4.5 之后才可以使用,但是 Task.Factory.StartNew 可以使用比 Task.Run 更多的参数,可以做到更多的定制.可以认 ...

  4. async,await与task.wait()或task.Result的区别

    你是否曾经与我一样不理解async,await与task.wait()或者task.Result的区别? 接下来,一个Demo让你看出他们之间的区别. static void Main(string[ ...

  5. Task 使用 Task以及Task.Factory都是在.Net 4引用的。Task跟Thread很类似,通过下面例子可以看到。

    static public void ThreadMain() { Thread t1 = new Thread(TaskWorker); t1.Start(3); } static public v ...

  6. Java 多线程实现接口Runnable和继承Thread区别(转)

    Java 多线程实现接口Runnable和继承Thread区别 Java中有两种实现多线程的方式.一是直接继承Thread类,二是实现Runnable接口.那么这两种实现多线程的方式在应用上有什么区别 ...

  7. Scheduler & Task & Worker & Thread & Request & Session & Connection of SQL Server

    MSSQL一直以来被人们认为简单.好学,但等到大家掌握了入门操作,深入理解起来又觉得非常的“拧巴”,尤其是对用惯了Oracle的同学来说,究其根本原因,无非是MSSQL引入和暴露了太多的概念.细节和理 ...

  8. 【.NET】- Task.Run 和 Task.Factory.StartNew 区别

    Task.Run 是在 dotnet framework 4.5 之后才可以使用, Task.Factory.StartNew 可以使用比 Task.Run 更多的参数,可以做到更多的定制. 可以认为 ...

  9. C# 的 Task、Thread、ThreadPool 之间有什么异同?

    Thread就是Thread,需要自己调度,适合长跑型的操作. ThreadPool是Thread基础上的一个线程池,目的是减少频繁创建线程的开销.线程很贵,要开新的stack,要增加CPU上下文切换 ...

随机推荐

  1. P1082 同余方程(拓展欧几里德)

    题目描述 求关于xx的同余方程 a x \equiv 1 \pmod {b}ax≡1(modb) 的最小正整数解. 输入输出格式 输入格式: 一行,包含两个正整数 a,ba,b,用一个空格隔开. 输出 ...

  2. Codeforces Round #560 (Div. 3)A-E

    A. Remainder output standard output You are given a huge decimal number consisting of nn digits. It ...

  3. 如何利用 docker 快速部署 Mysql 服务

    docker 基础教程不再多说,这里只着重讲如何使用 docker 部署 mysql 服务 docker 拉取 访问 dockerhub,搜索关键词 mysql,我这里选择 mysql-server, ...

  4. Jmeter简单操作 取样器 ,监听器

    1.创建线程组 1.2 (1)  线程组主要包含三个参数:线程数.准备时长(Ramp-Up Period(in seconds)).循环次数. (2) 线程数:虚拟用户数.一个虚拟用户占用一个进程或线 ...

  5. 剑指 Offer 48. 最长不含重复字符的子字符串

    题目描述 请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度. 示例1: 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 & ...

  6. LongAccumulator类的BUG——reset方法并不能保证初始值正确赋值

    LongAccumulator.reset方法并不能重置重置LongAccumulator的identity:初始值正确,使其恢复原来的初始值.当初始值为0是不会发生这个问题,而当我们设置初始值如1时 ...

  7. EMQ集群搭建实现高可用和负载均衡(百万级设备连接)

    一.EMQ集群搭建实现高可用和负载均衡 架构服务器规划 服务器IP 部署业务 作用 192.168.81.13 EMQTTD EMQ集群 192.168.81.22 EMQTTD EMQ集群 192. ...

  8. “未在本地计算机上注册“Microsoft.ACE.OLEDB.12.0”提供程序”的解决方案

    不论是连接Access数据库或是SQL Server数据库,"未在本地计算机上注册"Microsoft.ACE.OLEDB.12.0"提供程序."这个问题从Of ...

  9. elo system

    今天了解了一下游戏中的PVP模块的实现,大多数的游戏都使用到了ELO算法,刚开始的时候并不清楚这个算法是做什么的,对此开始大量查找有关于ELO算法的资源,功夫不负有心人,总算找到一些有用的资源了. 先 ...

  10. Linux:less and Aix:more

    在运维工作中,经常要查询应用日志,有Linux和Aix系统,个人感觉,Linux查询日志用less命令比较方便,Aix查询日志用more命令比较方便,在此总结一下两个命令的使用方法 AIX more命 ...