前一章我们提到了同步异步,多线程;在开始今天的文章之前,先来总结一下上一篇文章的内容,多线程的优点。

多线程有哪些优点呢,在这里通过代码依次来总结。

异步多线程的三大特点

1)同步方法卡界面,原因是主线程被占用;异步方法不卡界面,原因是计算交给了别的线程,主线程空闲

首先创建winfrom程序,建一个普通方法,然后异步、同步调用

        /// <summary>
/// 执行动作:耗时而已
/// </summary>
private void TestThread(string threadName)
{
Console.WriteLine("TestThread Start Name={2}当前线程的id:{0},当前时间为{1},", System.Threading.Thread.CurrentThread.ManagedThreadId, DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:fff"), threadName);
long sum = 0;
for (int i = 1; i < 999999999; i++)
{
sum += i;
}
//Thread.Sleep(1000);
Console.WriteLine("TestThread End Name={2}当前线程的id:{0},当前时间为{1},计算结果{3}", System.Threading.Thread.CurrentThread.ManagedThreadId, DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:fff"), threadName, sum);
}

同步调用

 private void btnSync_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("**********************btnSync_Click Start 主线程id {0}****************************************", Threa d.CurrentThread.ManagedThreadId); for (int i = 0; i < 5; i++)
{
string name = string.Format("btnSync_Click_{0}", i);
TestThread(name);
} watch.Stop();
Console.WriteLine("**********************btnSync_Click End 主线程id {0} {1}*******************************", Thread.Cur rentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

异步调用

   private void btnAsync_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("***********************btnAsync_Click Start 主线程id {0}**********************************", Thread.CurrentThread.ManagedThreadId); for (int i = 0; i < 5; i++)
{
string name = string.Format("btnAsync_Click_{0}", i);
Action act = () => this.TestThread(name);
act.BeginInvoke(null, null);
} watch.Stop();
Console.WriteLine("**********************btnAsync_Click End 主线程id {0} {1}************************************", Thread.CurrentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

通过运行,你会发现当同步调用时,整个winfrom界面是不能动的,你一动程序运行就会停止。

这是因为当使用同步方法时,主线程在运算,而移动winform界面的动作也是需要主线程操作。这就会形成卡顿或动不了的情况。

而异步调用就不会出现这种情况。上一章也讲过,异步调用时,方法体的运算实在其他线程中运行,而主线程会继续往下运行,很快就结束了运算,这个时候主线程是空闲的,所以就不会出现这总情况。

2) 同步方法慢,原因是只有一个线程计算;异步方法快,原因是多个线程同时计算,但是更消耗资源,不宜太多

通过运行结果也可以分析,同步方法主线程运行时间有十几秒,而异步方法也就六七秒。通过对CPU运行分析,就更能够理解了。

当不运行是,我么的CPU利用率保持在20%-30%之间。

但我们调用同步方法时,利用率到达了50%-60%之间

当我们调用异步方法时,CPU爆表,达到100%

所以说异步调用不能在程序中无限使用,不知道在哪里看过,据不官方的报道,程序的线程数在电脑cpu核数4倍左右会达到最佳。

简单来说异步多线程就是资源来换取时间。

3)异步多线程是无序的,启动顺序不确定、执行时间不确定、结束时间不确定

通过分析前面的运行结果可以看出,线程开始运行顺序是10、12、13、11、14,而结束是12、10、13、11、14。而且方法体运行时按理说应该是在循环体中从0-4依次开始,而异步开始的顺序是0、2、1、3、4,这些都是不确定的。

*************************************************************************************************************************************************************************************

前面总结了这么多,进入今天的主题,Thread(线程)。

其实线程(Thread)和前面的异步调用差不太多,关于概念的东西在前一篇文章中有阐述,这里就不说明了。那么我们怎么声明线程呢?

     private void btnThread_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("***********************btnThread_Click Start 主线程id {0}**********************************", Thread.CurrentThread.ManagedThreadId); List<Thread> threadList = new List<Thread>();
for (int i = 0; i < 5; i++)
{
string name = string.Format("btnThread_Click_{0}", i);
ThreadStart method = () => this.TestThread(name);
Thread thread = new Thread(method); thread.Start(); } watch.Stop();
Console.WriteLine("**********************btnThread_Click End 主线程id {0} {1}************************************", Thread.CurrentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

首先。就跟我们声明一个委托一样声明一个ThreadStart对象,然后把对象当变量的形式添加到线程中,最后开始线程。

注意:这样开始线程 默认前台线程,就是说程序退出后,计算任务会继续进行。

那么怎么改为后台线程呢,一般来说程序都退出了,就不需要进行计算任务了。这个时候就需要加一个属性

thread.IsBackground = true;// 后台线程:程序退出,计算立即结束

如果我们需要主线程等到其他线程结束后在结束,应该怎么做呢,也就是说希望主线程可以等待一段时间。

     private void btnThread_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("***********************btnThread_Click Start 主线程id {0}**********************************", Thread.CurrentThread.ManagedThreadId); List<Thread> threadList = new List<Thread>();
for (int i = 0; i < 5; i++)
{
string name = string.Format("btnThread_Click_{0}", i);
ThreadStart method = () => this.TestThread(name);
Thread thread = new Thread(method);//1 默认前台线程:程序退出后,计算任务会继续
thread.IsBackground = true;//2 后台线程:程序退出,计算立即结束
thread.Start();
threadList.Add(thread);
} foreach (Thread thread in threadList)
{
thread.Join();
} watch.Stop();
Console.WriteLine("**********************btnThread_Click End 主线程id {0} {1}************************************", Thread.CurrentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

我们需要声明一个Thread类型的List。没开始一个线程,就把线程加到list中,然后遍历,把每个线程Join(),因为这一句话是主线程运行的,也就是说主线程在等待其他线程结束后再接着运行。

Thread(线程)利于线程的回收,有点浪费资源,后来框架推出了池的概念。

也就是ThreadPool。就相当于添加了一个管理员,把线程都放在池子里,便于线程的管理。

  private void btnThreadPool_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("***********************btnThreadPool_Click Start 主线程id {0}**********************************", Thread.CurrentThread.ManagedThreadId); for (int i = 0; i < 5; i++)
{
string name = string.Format("btnThreadPool_Click_{0}", i);
WaitCallback method = t => this.TestThread(t.ToString());
ThreadPool.QueueUserWorkItem(method, name);
} watch.Stop();
Console.WriteLine("**********************btnThreadPool_Click End 主线程id {0} {1}************************************", Thread.CurrentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

我们只需要ThreadPool.QueueUserWorkItem(),接受一个WaitCallback类型的变量,和一个参数,这个参数会作用于WaitCallback。这就完成了线程的使用,简单了许多。。。但是如果我们也需要线程等待呢,该怎么做。

        private void btnThreadPool_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("***********************btnThreadPool_Click Start 主线程id {0}**********************************", Thread.CurrentThread.ManagedThreadId); ManualResetEvent mre = new ManualResetEvent(false);
WaitCallback method = t =>
{
this.TestThread(t.ToString());
mre.Set();//打开
};
ThreadPool.QueueUserWorkItem(method, "TestManualResetEvent"); Console.WriteLine("我们来干点别的。。。。");
Console.WriteLine("我们来干点别的。。。。");
Console.WriteLine("我们来干点别的。。。。");
Console.WriteLine("我们来干点别的。。。。");
mre.WaitOne(); watch.Stop();
Console.WriteLine("**********************btnThreadPool_Click End 主线程id {0} {1}************************************", Thread.CurrentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

只需要声明ManualResetEvent 并且设置为false;然后在委托中set打开一下,就行了。相应的ReSet就是关闭。关闭后如果不在打开就不会执行后面的内容

  ManualResetEvent mre = new ManualResetEvent(false);
new Action(() =>
{
Thread.Sleep(5000);
Console.WriteLine("委托的异步调用");
mre.Set();//打开
}).BeginInvoke(null, null); mre.WaitOne();
Console.WriteLine("12345");
mre.Reset();//关闭
new Action(() =>
{
Thread.Sleep(5000);
Console.WriteLine("委托的异步调用2");
// mre.Set();//打开
}).BeginInvoke(null, null);
mre.WaitOne();
Console.WriteLine("23456");

后面的打印23456就会停止打印,会一直处于等待状态。

******************************************************************************************************************************************************************************************************************************

后来在.net3.0时代推出了task概念。task,在这里讲几种创建方式,这里只是稍微涉及一下,在后面的章节中会细讲,

1)通过TaskFactory ,直接StartNew开始并创建一个线程。

  private void btnTask_Click(object sender, EventArgs e)
{
Stopwatch watch = new Stopwatch();
watch.Start();
Console.WriteLine();
Console.WriteLine("***********************btnTask_Click Start 主线程id {0}**********************************", Thread.CurrentThread.ManagedThreadId); TaskFactory taskFactory = new TaskFactory(); for (int i = 0; i < 5; i++)
{
string name = string.Format("btnAsync_Click_{0}", i);
Action act = () => this.TestThread(name);
taskFactory.StartNew(act);
} watch.Stop();
Console.WriteLine("**********************btnTask_Click End 主线程id {0} {1}************************************", Thread.CurrentThread.ManagedThreadId, watch.ElapsedMilliseconds);
Console.WriteLine();
}

2)建Task 对象,然后Start。

Task task = new Task(act);
task.Start();

3)直接对Task对象Run(),这样也可以对线程创建并开始;

Task task = Task.Run(act);

当然Task的创建不知这么几种,这里只是稍微涉及,在后面的章节会细讲^-^

未完待续。。。。。。。。。。

Threads(线程)(二)的更多相关文章

  1. 【Qt开发】事件循环与线程 二

    事件循环与线程 二 Qt 线程类 Qt对线程的支持已经有很多年了(发布于2000年九月22日的Qt2.2引入了QThread类),Qt 4.0版本的release则对其所有所支持平台默认地是对多线程支 ...

  2. C#中的线程二(Cotrol.BeginInvoke和Control.Invoke)

    C#中的线程二(Cotrol.BeginInvoke和Control.Invoke) 原文地址:http://www.cnblogs.com/whssunboy/archive/2007/06/07/ ...

  3. python并发编程之线程(二):死锁和递归锁&信号量&定时器&线程queue&事件evevt

    一 死锁现象与递归锁 进程也有死锁与递归锁,在进程那里忘记说了,放到这里一切说了额 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将 ...

  4. C#中的线程(二) 线程同步基础

    1.同步要领 下面的表格列展了.NET对协调或同步线程动作的可用的工具:                       简易阻止方法 构成 目的 Sleep 阻止给定的时间周期 Join 等待另一个线程 ...

  5. C#中的线程二(BeginInvoke和Invoke)

    近日,被Control的Invoke和BeginInvoke搞的头大,就查了些相关的资料,整理如下.感谢这篇文章对我的理解Invoke和BeginInvoke的真正含义 . (一)Control的In ...

  6. 【转】Qt事件循环与线程 二

    转自:http://blog.csdn.net/changsheng230/article/details/6153449 续上文:http://blog.csdn.net/changsheng230 ...

  7. 2.Perl 多线程:Threads(线程返回值)

    use warnings; use strict; use threads; sub TEST{ print "Hello, World!\n"; 'a'/); } #返回列表方法 ...

  8. java 线程二

    一.线程的优先级别 线程优先级别的使用范例: 1 package cn.galc.test; 2 3 public class TestThread6 { 4 public static void m ...

  9. Python线程二

    转自:https://www.cnblogs.com/chengd/articles/7770898.html 1. threading.Lock() import threading balance ...

  10. java基础学习总结——线程(二)

    一.线程的优先级别

随机推荐

  1. 为什么很多人使用#define而不是const定义常量

    众所周知,C语言一开始只有#define,C程序员用#define定义符号常量.但后来ANSI C加入了const限定符,而const应该比#define更好,为什么现在的C程序员还在大量使用#def ...

  2. OpenGL教程(0)——介绍

    OpenGL介绍 OpenGL,全称Open Graphics Library,是一个用C语言写的图形API.通俗地说,OpenGL用来绘制2D/3D图形.目前OpenGL的运用十分广泛,有许多用Op ...

  3. 一天搞定CSS:支持IE的Layout布局--16

    1.BFC和Layout区别: BFC和Layout的作用是一样的,只是对浏览器的支持不同而已. BFC- -标准浏览器所具有的 Layout- -IE浏览器所具有的 BFC详解地址:http://b ...

  4. Python抓取成都房价信息

    Python里scrapy爬虫 scrapy爬虫,正好最近成都房价涨的厉害,于是想着去网上抓抓成都最近的房价情况,顺便了解一下,毕竟咱是成都人,得看看这成都的房子我以后买的起不~ 话不多说,进入正题: ...

  5. Linux下如果忘记了Mysql的root密码该怎么办?

    下面十分简单的办法用来重置密码: 1.编辑MySQL配置文件my.cnf vi /etc/my.cnf #编辑文件,找到[mysqld],在下面添加一行skip-grant-tables [mysql ...

  6. UIPopoverPresentationController使用

    UIPopoverPresentationController是什么? iOS8.0之后引入的一个方便开发者创建带箭头的弹出控制器,类似qq消息页面点击右上角加号弹出的视图. UIPopoverPre ...

  7. jQuery – 鼠标经过(hover)事件的延时处理

    一.关于鼠标hover事件及延时 鼠标经过事件为web页面上非常常见的事件之一.简单的hover可以用CSS :hover伪类实现,复杂点的用js. 一般情况下,我们是不对鼠标hover事件进行延时处 ...

  8. VR虚拟现实技术在教育领域的前景展望

    VR虚拟现实技术在教育领域的前景展望 VR虚拟现实技术能迅速火起来,是基于它突破了人们对三维空间在时间与地域上的感知限制,以及市场需求愿景的升级.此技术可广泛地应用到城市规划.室内设计.工业仿真.古迹 ...

  9. 前端程序员的蜕变——JS的 event 对象属性、使用实例、兼容性处理(极大提高代码效率、减少代码量)

    下面讨论一下 js 中的 Event 对象,主要从以下三个方面详细的描述(点击标题可跳转到对应部分): 1.什么是event 2.怎么用event,用他该注意什么,几个简单实际应用 3.event在不 ...

  10. jQuery.merge( first, second )返回: Array

    jQuery.merge( first, second )返回: Array描述: 合并两个数组内容到第一个数组.first类型: Array第一个用于合并的数组,其中将会包含合并后的第二个数组的内容 ...