最常见的创建多线程的两种方式:继承Thread和实现Runnable接口
1.继承Thread类
class ThreadDemo extends Thread{
@Override
public void run() {
for (int i = 0; i < 1000; i++) {
System.out.println(Thread.currentThread().getName()+":"+i);
}
}
}
2.实现Runnable接口
class ThreadDemo implements Runnable{
@Override
public void run() {
for (int i = 0; i <1000; i++) {
System.out.println(Thread.currentThread().getName()+":"+i);
}
}
}
注意:对于使用继承Thread类的方法创建线程最大的局限就是不支持多继承,如果此时需要把子类变成线程类,使用继承Thread的方式就无法实现,就需要另一种方式,实现Runnable接口的方式,一边实现一边继承。这两种方式创建线程在工作时性质是一样的,没有本质区别,因为Thread类也实现了Runnable接口。
3.实现Callable接口
public class TestCallable {
public static void main(String[] args) {
ThreadDemo demo = new ThreadDemo();
//FutureTask, 实现类用于接收运算结果。
FutureTask<Integer> task = new FutureTask<>(demo);
new Thread(task).start();
try {
//接收线程运算后的结果
Integer integer = task.get();
System.out.println(integer);
} catch (InterruptedException |ExecutionException e) {
e.printStackTrace();
}
}
}
class ThreadDemo implements Callable<Integer>{
@Override
public Integer call() throws Exception {
int sum = 0;
for (int i = 1; i <=100; i++) {
sum=i+sum;
}
return sum;
}
}
实现 Callable 接口这种方式创建线程, 相较于实现 Runnable 接口的方式,主要区别在于方法可以有返回值,并且可以抛出异常
4.使用线程池的方式创建
线程池:提供了一个线程队列,队列中保存着所有等待状态的线程。避免了创建与销毁额外开销,提高了响应的速度
线程池的体系结构:
java.util.concurrent.Executor : 负责线程的使用与调度的根接口
|--ExecutorService 子接口: 线程池的主要接口
|--ThreadPoolExecutor 线程池的实现类
|--ScheduledExecutorService 子接口:负责线程的调度
|--ScheduledThreadPoolExecutor :继承 ThreadPoolExecutor, 实现 ScheduledExecutorService
工具类 : Executors
ExecutorService newFixedThreadPool() : 创建固定大小的线程池
ExecutorService newCachedThreadPool() : 缓存线程池,线程池的数量不固定,可以根据需求自动的更改数量。
ExecutorService newSingleThreadExecutor() : 创建单个线程池。线程池中只有一个线程
创建并使用
public class TestThreadPool {
public static void main(String[] args) throws Exception {
//创建线程池
ExecutorService pool = Executors.newFixedThreadPool(5);
ThreadPoolDemo t = new ThreadPoolDemo();
//为线程池中的线程分配任务
for (int i = 0; i < 10; i++) {
pool.submit(t);
}
//关闭线程池
pool.shutdown();
}
}
class ThreadPoolDemo implements Runnable{
private int i = 0;
@Override
public void run() {
while(i <= 100){
System.out.println(Thread.currentThread().getName() + " : " + i++);
}
}
}