C#线程学习笔记
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/18/Thread.html,保存方便资料查找
一、线程的介绍
进程(Process)是应用程序的实例要使用的资源的一个集合,每个应用程序都在各自的进程中运行来确保应用程序不受其他应用程序的影响。
线程是进程中基本执行单元, 一个进程中可以包含多个线程。在进程入口执行的第一个线程是一个进程的主线程,在.NET应用程序中,都是以Main()方法
作为程序的入口(线程是进程的执行单元,进程是线程的一个容器)。
二、线程调度和优先级
Windows之所以被称为抢占式多线程操作系统,是因为线程可以在任意时间被抢占,并调度另一个线程。
每个线程都分配了从0~31的一个优先级,系统首先把高优先级的线程分配给CPU执行。
Windows 支持7个相对线程优先级:Idle、Lowest、Below Normal、Normal、Above Normal、Highest和Time-Critical。Normal是默认的线程优先级,
然而在程序中可以通过设置Thread的Priority属性来改变线程的优先级,它的类型为ThreadPriority枚举类型:Lowest、BelowNormal、Normal、AboveNormal
和Highest,CLR为自己保留了 Idle和Time-Critical优先级。
枚举值列表如下:
成员名称 | 说明 |
Lowest | 可以将Thread置于其他优先级线程之后。 |
BelowNormal | 可以将Thread置于Normal优先级线程之后Lowest优先级线程之前。 |
Normal |
可以将Thread置于AboveNormal优先级线程之后BelowNormal优先级线程之前。 默认情况下,线程置于Normal优先级。 |
AboveNormal | 可以将Thread置于Highest优先级线程之后Normal优先级线程之前。 |
Highest | 可以将Thread置于其他优先级线程之前。 |
三、前台线程和后台线程
在.NET中线程分为前台线程和后台线程:
1、主线程是程序开始时就执行的,如果你需要再创建线程,那么创建的线程就是这个主线程的子线程,它是前台线程。
2、子线程可以是前台线程也可以是后台线程。
3、前台线程必须全部执行完,即使主线程关闭掉,这时进程仍然存活。
4、当所有前台线程停止运行时,CLR会强制结束仍在运行的任何后台线程,这些后台线程直接被终止,不会抛出异常。
5、前台线程与后台线程唯一的区别是后台线程不会阻止进程终止,可以在任何时候将前台线程修改为后台线程。
static void Main(string[] args)
{
ThreadType();
} /// <summary>
/// 前台线程与后台线程
/// </summary>
private static void ThreadType()
{
//创建一个新线程(默认为前台线程)
Thread backThread = new Thread(Worker)
{
//将线程更改为后台线程
IsBackground = true
}; //通过Start方法启动线程
backThread.Start(); //如果backThread是前台线程,则应用程序5秒后才终止。
//如果backThread是后台线程,则应用程序立即终止。
Console.WriteLine("It's from main thread.");
//Console.Read();
} private static void Worker()
{
//休息5秒
Thread.Sleep(5000);
Console.WriteLine("It's from worker thread.");
}
假如保留IsBackground = true;但又想继续执行Worker()方法的话,可以调用Thread.Join()方法来实现。Join()方法能保证主线程(前台线程)在异步线程
Thread(后台线程)运行结束后才会运行。
注1:一个线程在执行的过程中,可能调用另一个线程,前者可以称为调用线程,后者成为被调用线程。
注2:Thread.Join()方法的使用场景:调用线程挂起,等待被调用线程执行完毕后,继续执行。
注3:被调用线程执行Join方法,告诉调用线程,你先暂停,我执行完了,你再执行。从而保证了先后关系。
static void Main(string[] args)
{
ThreadStatusChange();
} /// <summary>
/// 线程状态之间的转换
/// </summary>
private static void ThreadStatusChange()
{
//创建一个新线程(默认为前台线程)
Thread backThread = new Thread(Worker)
{
//将线程更改为后台线程
IsBackground = true
}; //通过Start方法启动线程
backThread.Start(); //Join()方法能保证主线程(前台线程)在异步线程Thread(后台线程)运行结束后才会运行
backThread.Join(); Console.WriteLine("It's from main thread.");
Console.Read();
} private static void Worker()
{
//休息5秒
Thread.Sleep(5000);
Console.WriteLine("It's from worker thread.");
}
运行结果如下:
四、 Suspend和Resume方法
这两个方法在.NET Framework 1.0的时候就支持的方法,他们分别可以挂起线程及恢复挂起的线程,但在.NET Framework 2.0以后的版本中这两个方法都过时了。
MSDN的解释是这样:
警告:
不要使用Suspend和Resume方法来同步线程的活动。您无法知道挂起线程时它正在执行什么代码。如果您在安全权限评估期间挂起持有锁的线程,
则AppDomain中的其他线程可能被阻止。如果您在线程正在执行类构造函数时挂起它,则 AppDomain中尝试使用该类的其他线程将被阻止。这样很容易发生死锁。
static void Main(string[] args)
{
ThreadResume();
} /// <summary>
/// 线程恢复
/// </summary>
private static void ThreadResume()
{
Thread thread = new Thread(ThreadSuspend)
{
Name = "Thread1"
};
thread.Start();
Thread.Sleep(2000);
Console.WriteLine("Main Thread is running.");
//线程恢复
thread.Resume();
Console.Read();
} /// <summary>
/// 线程挂起
/// </summary>
private static void ThreadSuspend()
{
Console.WriteLine("Thread: {0} has been suspended.", Thread.CurrentThread.Name);
//将当前线程挂起
Thread.CurrentThread.Suspend();
Console.WriteLine("Thread: {0} has been resumed.", Thread.CurrentThread.Name);
}
在上面这段代码中Thread1线程是在主线程中恢复的,但当主线程发生异常时,这时候Thread1就会一直处于挂起状态,此时Thread1所使用的资源就不能释放
(除非强制终止进程),当其它的线程需要使用这快资源的时候, 很有可能就会发生死锁现象。
上面一段代码还存在一个隐患,假如把Thread.Sleep(2000);这段代码注释一下:
static void Main(string[] args)
{
ThreadResume();
} /// <summary>
/// 线程恢复
/// </summary>
private static void ThreadResume()
{
Thread thread = new Thread(ThreadSuspend)
{
Name = "Thread1"
};
thread.Start();
//Thread.Sleep(2000);
Console.WriteLine("Main Thread is running.");
//线程恢复
thread.Resume();
Console.Read();
} /// <summary>
/// 线程挂起
/// </summary>
private static void ThreadSuspend()
{
Console.WriteLine("Thread: {0} has been suspended.", Thread.CurrentThread.Name);
//将当前线程挂起
Thread.CurrentThread.Suspend();
Console.WriteLine("Thread: {0} has been resumed.", Thread.CurrentThread.Name);
}
这个时候,主线程因为跑(运行)得太快,做完自己的事情去唤醒Thread1时,此时Thread1还没有挂起,而此时唤醒Thread1就会出现异常了。
五、Abort和Interrupt方法
Abort方法和Interrupt都是用来终止线程的,但是两者还是有区别的:
1、它们抛出的异常不一样:Abort 方法抛出的异常是ThreadAbortException,Interrupt抛出的异常为ThreadInterruptedException。
2、调用Interrupt方法的线程之后可以被唤醒,然而调用Abort方法的线程就直接被终止不能被唤醒了。
下面演示Abort方法的使用:
static void Main(string[] args)
{
//ThreadType();
//ThreadStatusChange();
//ThreadResume();
ThreadAbort();
} /// <summary>
/// 线程中断(不可再唤醒)
/// </summary>
private static void ThreadAbort()
{
Thread threadAbort = new Thread(AbortMethod)
{
Name = "ThreadAbort"
};
threadAbort.Start();
Thread.Sleep(1000);
try
{
threadAbort.Abort();
}
catch
{
Console.WriteLine("1-> {0} exception happen in main thread.", Thread.CurrentThread.Name);
Console.WriteLine("2-> {0} status is:{1} in main thread.", Thread.CurrentThread.Name, Thread.CurrentThread.ThreadState);
}
finally
{
Console.WriteLine("3-> {0} status is:{1} in main thread.", threadAbort.Name, threadAbort.ThreadState);
}
threadAbort.Join();
Console.WriteLine("4-> {0} status is:{1}", threadAbort.Name, threadAbort.ThreadState);
Console.Read();
} /// <summary>
/// Abort方法
/// </summary>
private static void AbortMethod()
{
try
{
Thread.Sleep(5000);
}
catch (Exception e)
{
Console.WriteLine(e.GetType().Name);
Console.WriteLine("5-> {0} exception happen in abort thread.", Thread.CurrentThread.Name);
Console.WriteLine("6-> {0} status is:{1} in abort thread.", Thread.CurrentThread.Name, Thread.CurrentThread.ThreadState);
}
finally
{
Console.WriteLine("7-> {0} status is:{1} in abort thread.", Thread.CurrentThread.Name, Thread.CurrentThread.ThreadState);
}
}
运行结果如下:
从运行结果可以看出,调用Abort方法的线程引发的异常类型为ThreadAbortException,另外异常只会在调用Abort方法的线程中发生,而不会在主线程中抛出,
其次调用Abort方法后线程的状态不是立即改变为Aborted状态,而是从AbortRequested->Aborted。
下面演示Interrupt方法的使用:
static void Main(string[] args)
{
ThreadInterrupt();
} /// <summary>
/// 线程中断(可再唤醒)
/// </summary>
static void ThreadInterrupt()
{
Thread threadInterrupt = new Thread(InterruptMethod)
{
Name = "ThreadInterrupt"
};
threadInterrupt.Start();
threadInterrupt.Interrupt();
threadInterrupt.Join();
Console.WriteLine("1-> {0} status is:{1} ", threadInterrupt.Name, threadInterrupt.ThreadState);
Console.Read();
}
/// <summary>
/// Interrupt方法
/// </summary>
private static void InterruptMethod()
{
try
{
Thread.Sleep(5000);
}
catch (Exception e)
{
Console.WriteLine(e.GetType().Name);
Console.WriteLine("2-> {0} exception happen in interrupt thread.", Thread.CurrentThread.Name);
Console.WriteLine("3-> {0} status is:{1} in interrupt thread.", Thread.CurrentThread.Name, Thread.CurrentThread.ThreadState);
}
finally
{
Console.WriteLine("4-> {0} status is:{1} in interrupt thread.", Thread.CurrentThread.Name, Thread.CurrentThread.ThreadState);
}
}
运行结果如下:
从结果中可以得到,调用Interrupt方法抛出的异常为:ThreadInterruptException, 另外当调用Interrupt方法后线程的状态应该是中断的,但是从运行结果看,
此时的线程因为Join、Sleep方法而唤醒了线程。
为了进一步解释调用Interrupt方法的线程可以被唤醒, 我们可以在线程执行的方法中运用循环,如果线程可以唤醒,则输出结果中就一定会有循环的部分,
而调用Abort方法的线程则不会有循环的部分。
下面代码相信大家看后肯定会更加理解两个方法的区别:
static void Main(string[] args)
{
//ThreadType();
//ThreadStatusChange();
//ThreadResume();
//ThreadAbort();
//ThreadInterrupt();
ThreadWake();
} /// <summary>
/// 线程唤醒
/// </summary>
static void ThreadWake()
{
Thread threadWake = new Thread(WakeMethod);
threadWake.Start();
Thread.Sleep(100); threadWake.Interrupt();
Thread.Sleep(3000);
Console.WriteLine("1-> After finally block,the threadWake status is:{0}", threadWake.ThreadState);
Console.Read();
} /// <summary>
/// Wake方法
/// </summary>
private static void WakeMethod()
{
for (int i = 0; i < 4; i++)
{
try
{
Thread.Sleep(2000);
Console.WriteLine("2-> Thread is Running.");
}
catch (Exception ex)
{
if (ex != null)
{
Console.WriteLine("3-> Exception {0} throw.", ex.GetType().Name);
}
}
finally
{
Console.WriteLine("4-> Current thread status is:{0}", Thread.CurrentThread.ThreadState);
}
}
}
运行结果如下:
如果把上面的threadWake.Interrupt();改为threadWake.Abort(); 运行结果为:
六、简单线程的使用
其实在上面介绍前台线程和后台线程的时候已经通过ThreadStart委托创建一个线程了,此时已经实现了一个多线程的一个过程。
下面通过ParameterizedThreadStart委托的方式来实现多线程:
static void Main(string[] args)
{
ThreadTypeUseParameterized();
} /// <summary>
/// 前台线程与后台线程(使用ParameterizedThreadStart委托的方式来实现多线程)
/// </summary>
private static void ThreadTypeUseParameterized()
{
//创建一个新线程(默认为前台线程)
Thread backThread = new Thread(new ParameterizedThreadStart(Worker1)); //通过Start方法启动线程
backThread.Start(123); //如果backThread是前台线程,则应用程序5秒后才终止。
//如果backThread是后台线程,则应用程序立即终止。
Console.WriteLine("It's from main thread.");
} private static void Worker1(object parameter)
{
//休息5秒
Thread.Sleep(5000);
Console.WriteLine(parameter+" is from worker1 thread.");
Console.Read();
}
运行结果如下:
C#线程学习笔记的更多相关文章
- Linux进程线程学习笔记:运行新程序
Linux进程线程学习笔记:运行新程序 周银辉 在上一篇中我们说到,当启动一个新进程以后,新进程会复制父进程的大部份上下 ...
- C#线程学习笔记九:async & await入门二
一.异步方法返回类型 只能返回3种类型(void.Task和Task<T>). 1.1.void返回类型:调用方法执行异步方法,但又不需要做进一步的交互. class Program { ...
- C#线程学习笔记四:线程同步
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/21/ThreadsSynchronous.html,记录一下学习过程以备后续查用. ...
- C#线程学习笔记六:线程同步--信号量和互斥体
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/23/Mutex_And_Semaphore.html,记录一下学习过程以备后续查用. ...
- C#线程学习笔记五:线程同步--事件构造
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/23/Event_Constructor.html,记录一下学习过程以备后续查用. 前面讲的线 ...
- C#线程学习笔记三:线程池中的I/O线程
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/20/MultiThreads.html,记录一下学习过程以备后续查用. 一.I/O线 ...
- C#线程学习笔记二:线程池中的工作者线程
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/18/ThreadPool.html,记录一下学习过程以备后续查用. 一.线程池基础 首先,创 ...
- C#线程学习笔记一:线程基础
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/18/Thread.html,记录一下学习过程以备后续查用. 一.线程的介绍 进程(Proce ...
- 线程学习笔记 等待句柄和线程池(摘自https://blog.gkarch.com/threading/part2.html#manualresetevent)
//如果你的应用有很多线程,这些线程大部分时间都在阻塞,那么可以通过调用ThreadPool.RegisterWaitForSingleObject来减少资源消耗.这个方法接受一个委托,它会在向等待句 ...
随机推荐
- 【Pytest01】全网最全最新的Pytest框架快速入门
一.Pytest简介pytest是一个非常成熟的全功能的Python测试框架,主要有一下几个特点:1.简单灵活,容易上手,支持参数化2.能够支持简单的单元测试和复杂的功能测试,还可以用来做seleni ...
- [问题]java.sql.SQLException: The server time zone value 'Öйú±ê׼ʱ¼ä' is unrecognized...
java.sql.SQLException: The server time zone value 'Öйú±ê׼ʱ¼ä' is unrecognized... 这个问题一般是因为升级MSYQL ...
- Spring-Cloud-Netflix-Eureka注册中心
TOC 概述 eureka是Netflix的子模块之一,也是一个核心的模块 eureka里有2个组件: 一个是EurekaServer(一个独立的项目) 这个是用于定位服务以实现中间层服务器的负载平衡 ...
- P1004 方格取数(四维dp)
P1004 方格取数 思路如下 这题是看洛谷大佬的思路才写出来的,所以我会把大佬的思路展示如下: 1⃣️:我们可以找到一个叫思维dp的东西,dp[i][j][k][l],其中前两维表示一个人从原点出发 ...
- flink 一分钟入门篇
1. 业务说:“…… bulabula……,这个需求很简单,怎么实现我不管?” 面对霸气侧漏的业务需求,由于没有大数据知识储备,咱心里没底,咱也不敢问,咱也不敢说,只能静下来默默储备.默默寻觅解决方案 ...
- C语言:static关键字用法
参考博客:https://blog.csdn.net/guotianqing/article/details/79828100 看个例子: #include <stdio.h> void ...
- uni-app在线引入阿里字体图标库
第一步 在app.vue中引入阿里字体图标库 第二步 在任意页面使用就可以了 <view class="item" v-for="(value,index) in ...
- Java第五天,API常用类,静态(static)、集合(ArrayList)、日期(Date)、日历(Calendar)的使用方法
上文中我们学习到了Random随机数类和ArrayList<E>集合.这两个知识点都是经常用到的,那么除了这两个外,还有哪些知识点是我们所必须掌握的呢? static 使用static需要 ...
- django->基本操作和新建项目常用配置
一.安装django pip install django==2.1.5 -U #安装django/升级最新版本 二.创建.启动django项目 django-admin startproject m ...
- fiddler composer post请求
必加部分:Content-Type: application/json