java多线程创建的三种方法
一、线程的状态
新建状态:
使用 new 关键字和 Thread 类或其子类建立一个线程对象后,该线程对象就处于新建状态。它保持这个状态直到程序 start() 这个线程。
就绪状态:
当线程对象调用了start()方法之后,该线程就进入就绪状态。就绪状态的线程处于就绪队列中,要等待JVM里线程调度器的调度。
运行状态:
如果就绪状态的线程获取 CPU 资源,就可以执行 run(),此时线程便处于运行状态。处于运行状态的线程最为复杂,它可以变为阻塞状态、就绪状态和死亡状态。
阻塞状态:
如果一个线程执行了sleep(睡眠)、suspend(挂起)等方法,失去所占用资源之后,该线程就从运行状态进入阻塞状态。在睡眠时间已到或获得设备资源后可以重新进入就绪状态。可以分为三种:
等待阻塞:运行状态中的线程执行 wait() 方法,使线程进入到等待阻塞状态。
同步阻塞:线程在获取 synchronized 同步锁失败(因为同步锁被其他线程占用)。
其他阻塞:通过调用线程的 sleep() 或 join() 发出了 I/O 请求时,线程就会进入到阻塞状态。当sleep()状态超时,join() 等待线程终止或超时,或者 I/O 处理完毕,线程重新转入就绪状态。
死亡状态:一个运行状态的线程完成任务或者其他终止条件发生时,该线程就切换到终止状态。
二、创建方式:
通过实现 Runnable 接口;(最基本、最简单)
通过继承 Thread 类本身;(本质上也是实现了 Runnable 接口的一个实例)
通过 Callable 和 Future 创建线程。(可以获取线程执行后的返回值)
1.实现Runnable接口
/**
* 描述:
* 1.实现Runnable接口方式创建多线程
* @author 闲走天涯
* @create 2021/3/16 14:45
* 创建一个线程,最简单的方法是创建一个实现 Runnable 接口的类。
* 为了实现 Runnable,一个类只需要执行一个方法调用 run()
*/
public class TestRunnable implements Runnable{
private Integer num=100;
@Override
public void run() {
while (num <= 100) {
try {
Thread.sleep(300);
num--;
System.out.println("余票数:" + num);
if (num <= 0){break;}
}catch (Exception e){
System.out.println("异常,退出循环");
break;
}
}
}
public static void main(String[] args) throws Exception{
TestRunnable runnable = new TestRunnable();
new Thread(runnable).start();
Thread.sleep(250);
TestRunnable runnable2 = new TestRunnable();
new Thread(runnable2).start();
Thread.sleep(1000);
TestRunnable runnable3 = new TestRunnable();
new Thread(runnable3).start();
new Thread(()->System.out.println("java 8")).start();
}
}
2.继承Thread类创建多线程
Thread类的一些重要方法:
序号 | 方法 | 描述 |
---|---|---|
1 | public void start() | 使该线程开始执行;Java 虚拟机调用该线程的 run 方法。 |
2 | public void run() | 如果该线程是使用独立的 Runnable 运行对象构造的,则调用该 Runnable 对象的 run 方法;否则,该方法不执行任何操作并返回。 |
3 | public final void setName(String name) | 改变线程名称,使之与参数 name 相同。 |
4 | public final void setPriority(int priority) | 更改线程的优先级。 |
5 | public final void setDaemon(boolean on) | 将该线程标记为守护线程或用户线程。 |
6 | public final void join(long millisec) | 等待该线程终止的时间最长为 millis 毫秒。 |
7 | public void interrupt() | 中断线程。 |
8 | public final boolean isAlive() | 测试线程是否处于活动状态。 |
示例
/**
* 描述:
* 2.继承Thread类创建多线程
* @author 闲走天涯
* @create 2021/3/16 14:05
* 创建一个线程的第二种方法是创建一个新的类,该类继承 Thread 类,然后创建一个该类的实例。
* 继承类必须重写 run() 方法,该方法是新线程的入口点。它也必须调用 start() 方法才能执行。
* 该方法尽管被列为一种多线程实现方式,但是本质上也是实现了 Runnable 接口的一个实例。
*/
public class TestThread extends Thread{
private int num = 1;
/**
* 线程同步去整除10
* @return
*/
public Boolean doSth() {
try {
Thread.sleep(200);
num++;
System.out.println("num="+num);
if (num%10==0) {
return true;
} else {
return false;
}
} catch (InterruptedException e) {
System.out.println("报错:"+e.getMessage());
e.printStackTrace();
}
return false;
}
@Override
public void run() {
while (num < 60 * 5*100) {
if (doSth()) {
interrupt();
System.out.println("整除10");
break;
}
}
}
public static void main(String[] args) {
new TestThread().start();
new TestThread().start();
}
}
3.通过 Callable 和 Future 创建线程:实现Callable接口能够获取线程返回值,返回值被Future类封装,通过Future类的方法:get()可以获取线程的返回值。
/**
* @program: test
* @description: 3.通过 Callable 和 Future 创建线程:能够获取线程返回值
* @author: 闲走天涯
* @create: 2021-08-23 16:59
* 1. 创建 Callable 接口的实现类,并实现 call() 方法,该 call() 方法将作为线程执行体,并且有返回值。
* 2. 创建 Callable 实现类的实例,使用 FutureTask 类来包装 Callable 对象,该 FutureTask 对象封装了该 Callable 对象的 call() 方法的返回值。
* 3. 使用 FutureTask 对象作为 Thread 对象的 target 创建并启动新线程。
* 4. 调用 FutureTask 对象的 get() 方法来获得子线程执行结束后的返回值。
*/
public class CallableTest implements Callable<Integer> {
@Override
public Integer call() throws Exception {
int i=0;
for(;i<100;i++) {
Thread.currentThread().sleep(200);
System.out.println(Thread.currentThread().getName()+" "+i);
}
return i;
}
public static void main(String[] args) throws ExecutionException, InterruptedException {
//1.实例化CallableTest对象
CallableTest callableTest = new CallableTest();
//2.将CallableTest放入FutureTask
FutureTask<Integer> futureTask = new FutureTask<>(callableTest);
//3.将FutureTask放入Thread
Thread thread = new Thread(futureTask);
//4.启动线程
thread.start();
System.out.println("子线程的返回值:"+futureTask.get());
}
}