Java多线程总结笔记
实现多线程的方法
查看Thread类的源码,可以发现它实现了Runnable接口,然后在自己的run方法中调用了Runnable的run方法。这里其实就是静态代理这一设计模式,Thread是代理角色,而Runnable则是真实角色。在Thread的start方法中,会为我们开辟新的线程并执行其run方法,而Thread的run方法会调用它内部持有引用的Runnable的run方法。因此,我们有以下几种方法开辟新线程:
1. 重写Thread的run方法
通过重写Thread的run方法,不再传给它Runnable,通过其Runnable的run方法执行,而是在的run方法中直接执行相应的操作。
Thread thread = new Thread(){
@Override
public void run() {
//TODO
}
};
thread.start();
2. 通过Runnable的run方法
通过创建一个Runnable,通过静态代理的设计,将Runnable对象传给Thread,再通过Thread开辟新线程并做相应操作
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
//TODO
}
});
thread.start();
3. 通过Callable接口
通过Callable<T>
接口实现多线程,比较繁琐,优点是有返回值。
创建一个类实现Callable<T>
接口,其中泛型为返回的值的类型、然后实现Callable的call方法。
class Race implements Callable<Integer>{
@Override
public Integer call() throws Exception {
return 1000;
}
}
借助执行调度工具ExecutorService,创建对应类型线程池。获取Future对象(submit方法中的参数是具体实现类对象)。然后通过Future的get方法即可获取到返回值(此处要做异常处理,将异常throws了)。
get这里会等待线程中步骤执行完然后获取数据。
ExecutorService service = Executors.newFixedThreadPool(2);
Race race = new Race();
Future<Integer> result = service.submit(race);
int num = result.get();
调用ExecutorService的shutdown方法停止服务
service.shutdown();
线程的状态
线程有五大状态,分别是
- new:新建状态
- Runnable:就绪状态
- Running:运行状态
- Blocked:阻塞状态
- Dead:死亡状态
上面的图是线程的状态的示意图。
当我们创建线程后,它便进入了new(新建)状态。
在我们调用了它的start方法后,它便进入了Runnable(就绪)状态。
经过CPU的调度,它便会进入Running(运行)状态。
如果在运行后执行完毕,这个线程便会进入Dead(死亡)状态。
如果在运行过程中一些事情导致了线程的阻塞,它就会进入Blocked(阻塞)状态。阻塞解除后它便会重新进入Runnable(就绪状态)
线程的停止
1. 自然终止
当线程体自然执行完毕后,线程便会自然终止。
2. 外部干涉
外部干涉非常简单,比如下面这种方法
线程类中定义线程体使用标志
class Study implements Runnable{
private boolean flag = true;
@Override
public void run() {
//TODO
}
}
线程体内使用flag
class Study implements Runnable{
private boolean flag = true;
@Override
public void run() {
while (flag){
System.out.println("study thread");
}
}
}
提供对外方法改变该标识
class Study implements Runnable{
private boolean flag = true;
.</