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来减少资源消耗.这个方法接受一个委托,它会在向等待句 ...
随机推荐
- redis++:Redis的两种持久化 RDB 和 AOF
Redis持久化备份数据的方式有两种:RDB(Redis DataBase) . AOF(Append Only File). RDB 什么是RDB: 在指定时间间隔内,将内存中的数据集快照写入磁盘 ...
- 11g数据库使用DBUA升级Exadata数据库至12c
DB Name: GRPSource DB: 11.2.0.3 Target DB: 12.1.0.2 11g数据库使用DBUA升级Exadata数据库至12c前提条件: 0.需要11g源数据库fu ...
- [vijos1880]选课<树形dp>
题目链接:https://www.vijos.org/p/1180 这是一道树形dp的裸题,唯一的有意思的地方就是用到了多叉树转二叉树 然后本蒟蒻写这一道水题就是因为以前知道这个知识点但是没有怎么去实 ...
- Spring中常用注解的介绍
spring中使用注解时配置文件的写法: <?xml version="1.0" encoding="UTF-8"?> <span style ...
- Step by Step!教你如何在k3s集群上使用Traefik 2.x
本文来自边缘计算k3s社区 作者简介 Cello Spring,瑞士人.从电子起步,拥有电子工程学位.尔后开始关注计算机领域,在软件开发领域拥有多年的工作经验. Traefik是一个十分可靠的云原生动 ...
- 字符串学习笔记(三)---- StringBuilder
一.前言 StringBuilder是jdk1.5后出现的,而StringBuffer是jdk1.0就出现了,并且在功能上俩者并无太大区别.但为什么后来要添加一个StringBuilder呢?这是为了 ...
- 微信小程序wx:for隐藏遍历的最后一个元素
微信小程序开发时有时会需要从wx:for遍历的元素中选取最后一个来进行相关操作,以下方法以隐藏最后一个元素为例 index==list.length-1 通过获取列表的总长度减一来得到最后一个元素是最 ...
- python3.6 ubuntu部署nginx、 uwsgi、 django
ubuntu部署nginx. uwsgi. django 将项目上传到服务器 python manager.py runserver 0:80 在浏览器输入服务器的域名或者ip地址,访问成功. 安装u ...
- 2059 - Authentication plugin 'caching_sha2_password' cannot be loaded dlope
今天在mac上使用navicat连接mysql报错弄了一下午,各种查询踩坑,总算解决了. 即从mysql5.7版本之后,默认采用了caching_sha2_password验证方式,我用的mysql8 ...
- HBase协处理器加载的三种方式
本文主要给大家罗列了HBase协处理器加载的三种方式:Shell加载(动态).Api加载(动态).配置文件加载(静态).其中静态加载方式需要重启HBase. 我们假设我们已经有一个现成的需要加载的协处 ...