Java多线程入门类与接口

[TOC]

Java入门类和接口

Thread类和Runnable接口

Java中,JDK提供了Thread类和Runnable接口,我们有两种方法来实现自己的线程类。

  • 继承Thread类,并重写run方法。
  • 实现Runnable接口的run方法。

继承Thread

1
2
3
4
5
6
7
8
9
10
11
public class MyThread extends Thread {
public static void main(String[] args) {
MyThread myThread = new MyThread();
myThread.start();
}

@Override
public void run(){
System.out.println("我是线程: "+Thread.currentThread().getName());
}
}

注意:一定要调用start()方法该线程才算启动。

new一个Thread,线程进入新建状态;调用start()方法会启动一个线程并使线程进入就绪状态,当分配到CPU时间片就可以运行了。start()方法会执行线程的相应准备工作,然后自动执行run()方法的内容,这才是真正的多线程。而直接调用run()方法,会把run()方法当成一个main线程下的普通方法去执行,还是在主线程中。

多次调用start()方法会抛出java.lang.IllegalThreadStateException异常。

实现Runnable()接口

Runnable()接口源码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@FunctionalInterface
public interface Runnable {
/**
* When an object implementing interface <code>Runnable</code> is used
* to create a thread, starting the thread causes the object's
* <code>run</code> method to be called in that separately executing
* thread.
* <p>
* The general contract of the method <code>run</code> is that it may
* take any action whatsoever.
*
* @see java.lang.Thread#run()
*/
public abstract void run();
}

可以看到Runnable接口是一个函数式接口,这代表着可以使用函数式编程简化。

1
2
3
4
5
6
7
public class ThreadTest {
public static void main(String[] args) {
new Thread(() -> {
System.out.println("我是线程: " + Thread.currentThread().getName());
}).start();
}
}

不采用lambda要这样用:

1
2
3
4
5
6
7
8
9
10
11
12
13
public class ThreadTest {
public static class thread1 implements Runnable{

@Override
public void run() {
System.out.println("runnable "+Thread.currentThread().getName());
}
}

public static void main(String[] args) {
new Thread(new thread1()).start();
}
}

Thread类构造器

Thread类是Runnable接口的实现类,其构造方法源码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
/**
* Allocates a new {@code Thread} object. This constructor has the same
* effect as {@linkplain #Thread(ThreadGroup,Runnable,String) Thread}
* {@code (null, null, gname)}, where {@code gname} is a newly generated
* name. Automatically generated names are of the form
* {@code "Thread-"+}<i>n</i>, where <i>n</i> is an integer.
*/
public Thread() {
init(null, null, "Thread-" + nextThreadNum(), 0);
}

/**
* Allocates a new {@code Thread} object. This constructor has the same
* effect as {@linkplain #Thread(ThreadGroup,Runnable,String) Thread}
* {@code (null, target, gname)}, where {@code gname} is a newly generated
* name. Automatically generated names are of the form
* {@code "Thread-"+}<i>n</i>, where <i>n</i> is an integer.
*
* @param target
* the object whose {@code run} method is invoked when this thread
* is started. If {@code null}, this classes {@code run} method does
* nothing.
*/
public Thread(Runnable target) {
init(null, target, "Thread-" + nextThreadNum(), 0);
}
/**
* Allocates a new {@code Thread} object. This constructor has the same
* effect as {@linkplain #Thread(ThreadGroup,Runnable,String) Thread}
* {@code (null, target, name)}.
*
* @param target
* the object whose {@code run} method is invoked when this thread
* is started. If {@code null}, this thread's run method is invoked.
*
* @param name
* the name of the new thread
*/
public Thread(Runnable target, String name) {
init(null, target, name, 0);
}

可以看到Thread类的构造器会调用私有的init()方法来实现初始化。init方法的方法签名如下:

1
2
3
private void init(ThreadGroup g, Runnable target, String name,
long stackSize, AccessControlContext acc,
boolean inheritThreadLocals)

init方法参数的解释:

  • g:线程组,制定这个线程在哪个线程组下
  • target:指定要执行的任务;
  • name:线程的名字,多个线程的名字是可以重复的。如果不指定名字,会默认指定为"Thread-" + nextThreadNum()
  • acc:用于初始化私有变量inheritedAccessControlContext
  • inheritThreadLocals:可继承的ThreadLocalThread类中有两个私有属性来支持ThreadLocal

大多数情况下我们调用下面两种构造方法:

  • Thread(Runnable target)
  • Thread(Runnable target, String name)

Thread类的常用方法

  • currentThread():静态方法,返回对当前正在执行的线程对象的引用。
  • start():开始执行线程的方法,Java虚拟机会调用线程内的run方法。
  • yield():当前线程愿意让出对当前处理器的占用。要注意的是,就算当前线程调用了yield()方法,程序在调度的时候,也还有可能继续运行这个线程的。
  • sleep():静态方法,使当前线程睡眠指定时间。
  • join():使当前线程等待另一个线程执行完毕后再继续执行,内部调用的是Object类wait()方法。

Thread类与Runnable接口的比较

实现一个自定义线程类,可以有继承Thread类和实现Runnable接口两种方法。

  • 由于Java单继承、多实现的特点,Runnable接口使用更加灵活;
  • Thread类就是Runnable接口的实现;
  • Runnable接口更加符合面对对象,将线程单独进行对象的封装;
  • Runnable接口的出现,降低了线程对象与线程任务的耦合性;
  • 如果不需要使用Thread类的其他方法,使用Runnable接口更加轻量

综上,我们通常优先使用Runnable接口来自定义线程类。

CallableFutureFutureTask

通常我们使用RunnableThread来创建一个线程,但是run方法是没有返回值的。如果我们希望执行一个任务后返回一个值,我们可以使用JDK提供的Callable接口和Future类。这也是所谓的“异步”模型。

Callable接口

Callable接口与Runnable接口类似,同样是一个函数式接口,不同的是Callable接口有返回值且支持泛型

1
2
3
4
5
6
7
8
9
10
@FunctionalInterface
public interface Callable<V> {
/**
* Computes a result, or throws an exception if unable to do so.
*
* @return computed result
* @throws Exception if unable to compute a result
*/
V call() throws Exception;
}

Callable接口一般是配合线程池工具ExecutorSevice来使用的。ExecutorService可以使用submit方法来让一个Callable接口执行,它会返回一个Future,我们后续的程序可以通过这个Futureget方法得到结果。一个简单的demo如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//自定义Callable
public class ThreadTest implements Callable<Integer>{

@Override
public Integer call() throws Exception {
Thread.sleep(1000);
return 2;
}

public static void main(String[] args) throws ExecutionException, InterruptedException {
//使用
ExecutorService executor = Executors.newCachedThreadPool();
//传入一个Callable对象
Future<Integer> result = executor.submit(new ThreadTest());
//注意调用get方法会阻塞当前线程,直到得到结果
//实际编码中建议使用可以设置超时时间的重载get方法
System.out.println(result.get());
}
}

Future接口

Future接口的源码如下:

1
2
3
4
5
6
7
8
public interface Future<V> {
boolean cancel(boolean mayInterruptIfRunning);
boolean isCancelled();
boolean isDone();
V get() throws InterruptedException, ExecutionException;
V get(long timeout, TimeUnit unit)
throws InterruptedException, ExecutionException, TimeoutException;
}

cancel方法是试图取消一个线程的执行。注意是试图取消,不一定取消成功。因为任务可能已完成、已取消或者不可取消。最后返回是否取消成功。

参数mayInterruptIfRunning代表是否采用中断的方式取消线程执行。

所以有时候,为了让线程有能够取消的可能,就会使用Callable代替Runnable。若只是为了可取消性,而不需要结果,可以声明Future<?>形式类型,并返回null作为结果。

FutureTask

Future接口有一个实现类FutureTask,这个类实现了RunnableFuture接口,而RunnableFuture接口同时继承了Runnable接口和Future接口。

RunnableFuture接口源码如下:

1
2
3
4
5
6
7
public interface RunnableFuture<V> extends Runnable, Future<V> {
/**
* Sets this Future to the result of its computation
* unless it has been cancelled.
*/
void run();
}

Future只是一个接口,里面的cancelgetisDone等方法自己实现会很复杂,所以JDk提供了一个FutureTask类。使用示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public class ThreadTest implements Callable<Integer> {

@Override
public Integer call() throws Exception {
Thread.sleep(1000);
return 2;
}

public static void main(String[] args) throws ExecutionException, InterruptedException {
ExecutorService executor = Executors.newCachedThreadPool();
FutureTask<Integer> futureTask = new FutureTask<>(new ThreadTest());
//这里实际调用的是submit(Runnable task)方法
executor.submit(futureTask);
System.out.println(futureTask.get());
}
}

这里调用submit方法是没有返回值的。这里实际上调用的是submit(Runnable task)方法,而上面那个demo中,调用的是submit(Callable<T> task)方法。

这里是使用futureTask直接get取值,而上面Demo中是使用返回的Future去取值。

在高并发的情景下,有可能CallableFutureTask会创建多次。FutureTask能确保在高并发下任务只被执行一次。

FutureTask的几个状态

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/**
*state可能的状态转变路径如下:
* Possible state transitions:
* NEW -> COMPLETING -> NORMAL
* NEW -> COMPLETING -> EXCEPTIONAL
* NEW -> CANCELLED
* NEW -> INTERRUPTING -> INTERRUPTED
*/
private volatile int state;
private static final int NEW = 0;
private static final int COMPLETING = 1;
private static final int NORMAL = 2;
private static final int EXCEPTIONAL = 3;
private static final int CANCELLED = 4;
private static final int INTERRUPTING = 5;
private static final int INTERRUPTED = 6;

state表示任务的运行状态。初始状态为NEW。运行状态只会在setsetExceptioncancel方法中终止。COMPLETINGINTERRUPTING是任务完成后的瞬时状态。

-------------本文结束感谢您的阅读-------------