并发-线程基础
一、线程简介
什么是进程
进程是操作系统进行资源分配的基本单位。
进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动。
进程可视为一个正在运行的程序。它是系统运行程序的基本单位,因此进程是动态的。
什么是线程
线程是操作系统进行调度的基本单位。
线程也叫轻量级进程(Light Weight Process),在一个进程里可以创建多个线程,这些线程都拥有各自的计数器、堆栈和局部变量等属性,并且能够访问共享的内存变量。
进程和线程的区别
- 一个程序至少有一个进程,一个进程至少有一个线程。
- 线程比进程划分更细,所以执行开销更小,并发性更高。
- 进程是一个实体,拥有独立的资源;而同一个进程中的多个线程共享进程的资源。
二、线程基本用法
线程(Thread
)基本方法清单:
方法 | 描述 |
---|---|
run |
线程的执行实体。 |
start |
线程的启动方法。 |
currentThread |
返回对当前正在执行的线程对象的引用。 |
setName |
设置线程名称。 |
getName |
获取线程名称。 |
setPriority |
设置线程优先级。Java 中的线程优先级的范围是 [1,10],一般来说,高优先级的线程在运行时会具有优先权。可以通过 thread.setPriority(Thread.MAX_PRIORITY) 的方式设置,默认优先级为 5。 |
getPriority |
获取线程优先级。 |
setDaemon |
设置线程为守护线程。 |
isDaemon |
判断线程是否为守护线程。 |
isAlive |
判断线程是否启动。 |
interrupt |
中断另一个线程的运行状态。 |
interrupted |
测试当前线程是否已被中断。通过此方法可以清除线程的中断状态。换句话说,如果要连续调用此方法两次,则第二次调用将返回 false(除非当前线程在第一次调用清除其中断状态之后且在第二次调用检查其状态之前再次中断)。 |
join |
可以使一个线程强制运行,线程强制运行期间,其他线程无法运行,必须等待此线程完成之后才可以继续执行。 |
Thread.sleep |
静态方法。将当前正在执行的线程休眠。 |
Thread.yield |
静态方法。将当前正在执行的线程暂停,让其他线程执行。 |
创建线程
创建线程有三种方式:
- 继承
Thread
类 - 实现
Runnable
接口 - 实现
Callable
接口
继承 Thread 类
通过继承 Thread
类创建线程的步骤:
- 定义
Thread
类的子类,并覆写该类的run
方法。run
方法的方法体就代表了线程要完成的任务,因此把run
方法称为执行体。 - 创建
Thread
子类的实例,即创建了线程对象。 - 调用线程对象的
start
方法来启动该线程。
实现 Runnable 接口
实现 Runnable
接口优于继承 Thread
类,因为:
- Java 不支持多重继承,所有的类都只允许继承一个父类,但可以实现多个接口。如果继承了
Thread
类就无法继承其它类,这不利于扩展。 - 类可能只要求可执行就行,继承整个
Thread
类开销过大。
通过实现 Runnable
接口创建线程的步骤:
- 定义
Runnable
接口的实现类,并覆写该接口的run
方法。该run
方法的方法体同样是该线程的线程执行体。 - 创建
Runnable
实现类的实例,并以此实例作为Thread
的 target 来创建Thread
对象,该Thread
对象才是真正的线程对象。 - 调用线程对象的
start
方法来启动该线程。
实现 Callable 接口
继承 Thread 类和实现 Runnable 接口这两种创建线程的方式都没有返回值。所以,线程执行完后,无法得到执行结果。但如果期望得到执行结果该怎么做?
为了解决这个问题,Java 1.5 后,提供了
Callable
接口和Future
接口,通过它们,可以在线程执行结束后,返回执行结果。
通过实现 Callable
接口创建线程的步骤:
- 创建
Callable
接口的实现类,并实现call
方法。该call
方法将作为线程执行体,并且有返回值。 - 创建
Callable
实现类的实例,使用FutureTask
类来包装Callable
对象,该FutureTask
对象封装了该Callable
对象的call
方法的返回值。 - 使用
FutureTask
对象作为Thread
对象的 target 创建并启动新线程。 - 调用
FutureTask
对象的get
方法来获得线程执行结束后的返回值。
start
和 run
方法有什么区别
run
方法是线程的执行体。start
方法会启动线程,然后 JVM 会让这个线程去执行run
方法。
可以直接调用 Thread
类的 run
方法么
- 可以。但是如果直接调用
Thread
的run
方法,它的行为就会和普通的方法一样。 - 为了在新的线程中执行我们的代码,必须使用
Thread
的start
方法。
线程休眠
使用 Thread.sleep
方法可以使得当前正在执行的线程进入休眠状态。
使用 Thread.sleep
需要向其传入一个整数值,这个值表示线程将要休眠的毫秒数。
Thread.sleep
方法可能会抛出 InterruptedException
,因为异常不能跨线程传播回 main
中,因此必须在本地进行处理。线程中抛出的其它异常也同样需要在本地进行处理。
线程礼让
Thread.yield
方法的调用声明了当前线程已经完成了生命周期中最重要的部分,可以切换给其它线程来执行 。
该方法只是对线程调度器的一个建议,而且也只是建议具有相同优先级的其它线程可以运行。
终止线程
Thread
中的stop
方法有缺陷,已废弃。使用
Thread.stop
停止线程会导致它解锁所有已锁定的监视器(由于未经检查的ThreadDeath
异常会在堆栈中传播,这是自然的结果)。 如果先前由这些监视器保护的任何对象处于不一致状态,则损坏的对象将对其他线程可见,从而可能导致任意行为。Thread.stop
的许多用法应由仅修改某些变量以指示目标线程应停止运行的代码代替。 目标线程应定期检查此变量,如果该变量指示要停止运行,则应按有序方式从其运行方法返回。如果目标线程等待很长时间(例如,在条件变量上),则应使用中断方法来中断等待。
安全地终止线程有两种方法:
- 定义
volatile
标志位,在run
方法中使用标志位控制线程终止 - 使用
interrupt
方法和Thread.interrupted
方法配合使用来控制线程终止
守护线程
什么是守护线程?
- 守护线程(Daemon Thread)是在后台执行并且不会阻止 JVM 终止的线程。当所有非守护线程结束时,程序也就终止,同时会杀死所有守护线程。
- 与守护线程(Daemon Thread)相反的,叫用户线程(User Thread),也就是非守护线程。
为什么需要守护线程?
- 守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。典型的应用就是垃圾回收器。
如何使用守护线程?
- 可以使用
isDaemon
方法判断线程是否为守护线程。 - 可以使用
setDaemon
方法设置线程为守护线程。 - 正在运行的用户线程无法设置为守护线程,所以
setDaemon
必须在thread.start
方法之前设置,否则会抛出llegalThreadStateException
异常; - 一个守护线程创建的子线程依然是守护线程。
- 不要认为所有的应用都可以分配给守护线程来进行服务,比如读写操作或者计算逻辑。
sleep、yield、join 方法有什么区别
yield
方法yield
方法会 让线程从Running
状态转入Runnable
状态。- 当调用了
yield
方法后,只有与当前线程相同或更高优先级的Runnable
状态线程才会获得执行的机会。
sleep
方法sleep
方法会 让线程从Running
状态转入Waiting
状态。sleep
方法需要指定等待的时间,超过等待时间后,JVM 会将线程从Waiting
状态转入Runnable
状态。- 当调用了
sleep
方法后,无论什么优先级的线程都可以得到执行机会。 sleep
方法不会释放“锁标志”,也就是说如果有synchronized
同步块,其他线程仍然不能访问共享数据。
join
join
方法会 让线程从Running
状态转入Waiting
状态。- 当调用了
join
方法后,当前线程必须等待调用join
方法的线程结束后才能继续执行。
为什么 sleep 和 yield 方法是静态的
Thread
类的 sleep
和 yield
方法将处理 Running
状态的线程。
所以在其他处于非 Running
状态的线程上执行这两个方法是没有意义的。这就是为什么这些方法是静态的。它们可以在当前正在执行的线程中工作,并避免程序员错误的认为可以在其他非运行线程调用这些方法。
Java 线程是否按照线程优先级严格执行
即使设置了线程的优先级,也无法保证高优先级的线程一定先执行。
原因在于线程优先级依赖于操作系统的支持,然而,不同的操作系统支持的线程优先级并不相同,不能很好的和 Java 中线程优先级一一对应。
三、线程间通信
当多个线程可以一起工作去解决某个问题时,如果某些部分必须在其它部分之前完成,那么就需要对线程进行协调
wait/notify/notifyAll
wait
-wait
会自动释放当前线程占有的对象锁,并请求操作系统挂起当前线程,让线程从Running
状态转入Waiting
状态,等待notify
/notifyAll
来唤醒。如果没有释放锁,那么其它线程就无法进入对象的同步方法或者同步控制块中,那么就无法执行notify
或者notifyAll
来唤醒挂起的线程,造成死锁。notify
- 唤醒一个正在Waiting
状态的线程,并让它拿到对象锁,具体唤醒哪一个线程由 JVM 控制 。notifyAll
- 唤醒所有正在Waiting
状态的线程,接下来它们需要竞争对象锁。
注意:
wait
、notify
、notifyAll
都是Object
类中的方法,而非Thread
。wait
、notify
、notifyAll
只能用在synchronized
方法或者synchronized
代码块中使用,否则会在运行时抛出IllegalMonitorStateException
。为什么
wait
、notify
、notifyAll
不定义在Thread
中?为什么wait
、notify
、notifyAll
要配合synchronized
使用?首先,需要了解几个基本知识点:
- 每一个 Java 对象都有一个与之对应的 监视器(monitor)
- 每一个监视器里面都有一个 对象锁 、一个 等待队列、一个 同步队列
了解了以上概念,我们回过头来理解前面两个问题。
为什么这几个方法不定义在
Thread
中?由于每个对象都拥有对象锁,让当前线程等待某个对象锁,自然应该基于这个对象(
Object
)来操作,而非使用当前线程(Thread
)来操作。因为当前线程可能会等待多个线程的锁,如果基于线程(Thread
)来操作,就非常复杂了。为什么
wait
、notify
、notifyAll
要配合synchronized
使用?如果调用某个对象的
wait
方法,当前线程必须拥有这个对象的对象锁,因此调用wait
方法必须在synchronized
方法和synchronized
代码块中。
join
在线程操作中,可以使用 join
方法让一个线程强制运行,线程强制运行期间,其他线程无法运行,必须等待此线程完成之后才可以继续执行。
管道
管道输入/输出流和普通的文件输入/输出流或者网络输入/输出流不同之处在于,它主要用于线程之间的数据传输,而传输的媒介为内存。 管道输入/输出流主要包括了如下 4 种具体实现:PipedOutputStream
、PipedInputStream
、PipedReader
和 PipedWriter
,前两种面向字节,而后两种面向字符。
四、线程状态
java.lang.Thread.State
中定义了 6 种不同的线程状态,在给定的一个时刻,线程只能处于其中的一个状态。
以下是各状态的说明,以及状态间的联系:
-
新建(New) - 尚未调用
start
方法的线程处于此状态。此状态意味着:创建的线程尚未启动。 -
就绪(Runnable) - 已经调用了
start
方法的线程处于此状态。此状态意味着:线程已经在 JVM 中运行。但是在操作系统层面,它可能处于运行状态,也可能等待资源调度(例如处理器资源),资源调度完成就进入运行状态。所以该状态的可运行是指可以被运行,具体有没有运行要看底层操作系统的资源调度。 -
阻塞(Blocked) - 请求获取 monitor lock 从而进入
synchronized
函数或者代码块,但是其它线程已经占用了该 monitor lock,所以处于阻塞状态。要结束该状态进入Runnable
,从而需要其他线程释放 monitor lock。此状态意味着:线程处于被阻塞状态。 -
等待(Waiting) - 此状态意味着:线程等待被其他线程显式地唤醒。 阻塞和等待的区别在于,阻塞是被动的,它是在等待获取 monitor lock。而等待是主动的,通过调用
Object.wait
等方法进入。进入方法 退出方法 没有设置 Timeout 参数的 Object.wait
方法Object.notify
/Object.notifyAll
没有设置 Timeout 参数的 Thread.join
方法被调用的线程执行完毕 LockSupport.park
方法LockSupport.unpark
-
定时等待(Timed waiting) - 此状态意味着:无需等待其它线程显式地唤醒,在一定时间之后会被系统自动唤醒。
进入方法 退出方法 Thread.sleep
方法时间结束 设置了 Timeout 参数的 Object.wait
方法时间结束 / Object.notify
/Object.notifyAll
设置了 Timeout 参数的 Thread.join
方法时间结束 / 被调用的线程执行完毕 LockSupport.parkNanos
方法LockSupport.unpark
LockSupport.parkUntil
方法LockSupport.unpark
-
终止(Terminated) - 线程
run
方法执行结束,或者因异常退出了run
方法。此状态意味着:线程结束了生命周期。