1、JAVA多线程实现的四种方式
- 继承Thread类
- 实现Runnable接口
- 实现Callable接口通过FutureTask包装器来创建Thread线程
- 使用ExecutorService、Callable、Future实现有返回结果的多线程
2、继承Thread类创建线程
- Thread类本质上是实现了Runnable接口的一个实例
- 代表一个线程的实例
- 启动线程的唯一方法就是通过Thread类的start()实例方法
- start()方法是一个native方法,它将启动一个新线程,并执行run()方法
3、实现Runnable接口创建线程
- 实现Runnable接口
- 继承 Thread 有弊端,java是单继承的
- 启动MyThread,需要首先实例化一个Thread,并传入自己的MyThread实例
4、实现Callable接口通过FutureTask包装器来创建Thread线程
- Callable接口(也只有一个方法)定义如下:
- 用法如下:
5、使用ExecutorService、Callable、Future实现有返回结果的线程
- 可返回值的任务必须实现Callable接口
- 无返回值的任务必须实现Runnable接口
- 执行Callable任务后,可以获取一个Future的对象,在该对象上调用get就可以获取到Callable任务返回的Object
- 注意:get方法是阻塞的,即:线程无返回结果,get方法会一直等待
/**
* 有返回值的线程
*/
@SuppressWarnings("unchecked")
public class Test {
public static void main(String[] args) throws ExecutionException,
InterruptedException {
System.out.println("----程序开始运行----");
Date date1 = new Date();
int taskSize = 5;
// 创建一个线程池
ExecutorService pool = Executors.newFixedThreadPool(taskSize);
// 创建多个有返回值的任务
List<Future> list = new ArrayList<Future>();
for (int i = 0; i < taskSize; i++) {
Callable c = new MyCallable(i + " ");
// 执行任务并获取Future对象
Future f = pool.submit(c);
// System.out.println(">>>" + f.get().toString());
list.add(f);
}
// 关闭线程池
pool.shutdown();
// 获取所有并发任务的运行结果
for (Future f : list) {
// 从Future对象上获取任务的返回值,并输出到控制台
System.out.println(">>>" + f.get().toString());
}
Date date2 = new Date();
System.out.println("----程序结束运行----,程序运行时间【"
+ (date2.getTime() - date1.getTime()) + "毫秒】");
}
}
class MyCallable implements Callable<Object> {
private String taskNum;
MyCallable(String taskNum) {
this.taskNum = taskNum;
}
public Object call() throws Exception {
System.out.println(">>>" + taskNum + "任务启动");
Date dateTmp1 = new Date();
Thread.sleep(1000);
Date dateTmp2 = new Date();
long time = dateTmp2.getTime() - dateTmp1.getTime();
System.out.println(">>>" + taskNum + "任务终止");
return taskNum + "任务返回运行结果,当前任务时间【" + time + "毫秒】";
}
}