首页 > 编程语言 >Java多线程并发01——线程的创建与终止,几种方式介绍

Java多线程并发01——线程的创建与终止,几种方式介绍

时间:2023-02-02 16:31:30浏览次数:41  
标签:01 Java thread Thread Callable 线程 run 多线程 public

线程的创建方式

在 Java 中,用户常用的主动创建线程的方式有三种,分别是 继承 Thread 类实现 Runnable 接口通过Callable<Class> 和 Future

继承 Thread 类

  • 定义 Thread 类的子类,并重写该类的 run 方法;
  • 调用线程对象的 start() 方法来启动该线程。

通过继承 Thread 实现的线程类,多个线程间无法共享线程类的实例变量(需要创建不同 Thread 对象)。


/**
* 通过继承Thread实现线程
*/
public class MyThread extends Thread {
public void run() {
System.out.println("MyThread.run()");
}
}

MyThread myThread = new MyThread();
myThread.start();

实现 Runnable 接口

  • 如果自己的类已经 extends 另一个类,就无法直接 extends Thread,此时,可以实现一个 Runnable 接口;
  • 调用线程对象的start()方法来启动该线程。


/**
* 通过实现Runnable接口实现的线程类
*/
public class RunnableTest implements Runnable {
@Override
public void run() {
System.out.println("RunnableTest.run()");
}

public static void main(String[] args) {
RunnableTest runnableTest = new RunnableTest() ;
Thread thread = new Thread(runnableTest);
thread.start();
}
}

通过 Callable、Future

从 Thread 和 Runnable 两种方式可以看出,两种方式都不支持返回值,且不能声明抛出异常

而 Callable 接口则实现了此两点,Callable 接口如同 Runable 接口的升级版,其提供的 call() 方法将作为线程的执行体,同时允许有返回值。

但是 Callable 对象不能直接作为 Thread 对象的 target,我们可以使用 FutureTask 类来包装 Callable 对象,该 FutureTask 对象封装了该 Callable 对象的 call() 方法的返回值。


import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;

public class CallableTest {
public static void main(String[] args) {
CallableTest callableTest = new CallableTest() ;
//因为Callable接口是函数式接口,可以使用Lambda表达式
FutureTask<String> task = new FutureTask<Integer>((Callable<String>)()->{
System.out.println("FutureTask and Callable");
return "hello word";
});

try{
System.out.println("子线程返回值 : " + task.get());
} catch (Exception e){
e.printStackTrace();
}
}
}

线程的终止方式

线程除了正常结束外,还可以通过特定方式终止线程,终止线程常用的方式有以下三种:使用退出标志退出线程、** Interrupt 方法结束线程stop 方法终止线程**。

使用退出标志退出线程

最常使用的方式其实现方式是:定义一个 boolean 型的标志位,在线程的 run() 方法中根据这个标志位是 true 还是 false 来判断是否退出,这种情况一般是将任务放在 run() 方法中的一个 while 循环中执行的。


public class ThreadSafe extends Thread {
public volatile boolean exit = false;
public void run() {
while (!exit){
//do work
}
}

public static void main(String[] args) throws Exception {
ThreadFlag thread = new ThreadFlag();
thread.start();
sleep(5000); // 主线程延迟5秒
thread.exit = true; // 终止线程thread
thread.join();
System.out.println("线程退出!");
}
}

Interrupt 方法结束线程

使用 interrupt() 方法来中断线程有两种情况:

  1. 线程处于阻塞状态。如使用了 sleep,同步锁的 wait,socket 中的 receiver,accept 等方法时,会使线程处于阻塞状态。​​使用 interrupt 方法结束线程的时候,一定要先捕获 InterruptedException 异常之后通过 break 来跳出循环,才能正常结束 run 方法。​


public class ThreadInterrupt extends Thread {  
public void run() {
try {
sleep(50000); // 延迟50秒
}
catch (InterruptedException e) {
System.out.println(e.getMessage());
}
}
public static void main(String[] args) throws Exception {
Thread thread = new ThreadInterrupt();
thread.start();
System.out.println("在50秒之内按任意键中断线程!");
System.in.read();
thread.interrupt();
thread.join();
System.out.println("线程已经退出!");
}
}
  1. 线程未处于阻塞状态。使用 isInterrupted() 判断线程的中断标志来退出循环。当使用 interrupt() 方法时,中断标志就会置 true,和使用自定义的标志来控制循环是一样的道理。

Java多线程并发01——线程的创建与终止,几种方式介绍_System

public class ThreadSafe extends Thread {
public void run() {
while (!isInterrupted()) { //非阻塞过程中通过判断中断标志来退出
try {
Thread.sleep(5*1000);//阻塞过程捕获中断异常来退出
} catch (InterruptedException e) {
e.printStackTrace();
break;//捕获到异常之后,执行 break 跳出循环
}
}
}
}

stop 方法终止线程

使用 stop 方法可以强行终止正在运行或挂起的线程。我们可以使用如下的代码来终止线程:


thread.stop();

采用 stop 是不安全的,主要影响点如下:

  1. thread.stop() 调用之后,创建子线程的线程就会抛出 ThreadDeatherror 的错误;
  2. 调用 stop 会释放子线程所持有的所有锁。导致了该线程所持有的所有锁的突然释放(不可控制),那么被保护数据就有可能呈现不一致性。

总结

  • 线程创建:推荐使用 Runnable 或者 Callable 方式创建线程,相比继承,接口实现可以更加灵活,不会受限于Java的单继承机制。
  • 线程终止:线程终止推荐使用 标志位 或 Interrupt 方式终止,stop 方式对线程不安全,易导致数据不一致。

标签:01,Java,thread,Thread,Callable,线程,run,多线程,public
From: https://blog.51cto.com/u_13146445/6033741

相关文章

  • Java多线程并发02——线程的生命周期与常用方法
    线程生命周期一个线程不是被创建了马上就开始执行,也不是一直处于执行状态。在线程的整个生命周期中会经历新建(New)、就绪(Runnable)、运行(Running)、阻塞(Blocked)和销毁(Terminate......
  • java Set和List的区别
    1、重复。Set不允许重复插入。2、插入顺序。Set不能保证插入顺序。3、null元素。4、实现类。list方法常用的实现类:ArrayList、LinkedList和Vector。Set:HashSet、Lin......
  • java的BigDecimal比较大小
    BigDecimala=newBigDecimal(10);BigDecimalb=newBigDecimal(2);if(a.compareTo(b)==0){System.out.println("a等于b");}if(a.compareTo(b)==1){......
  • Java里面为什么搞了双重检查锁,写完这篇文章终于真相大白了
    双重检查锁定与延迟初始化在java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化。此时程序员可能会采用延迟初始化。但要正......
  • Java里面为什么搞了双重检查锁,写完这篇文章终于真相大白了
    双重检查锁定与延迟初始化在java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化。此时程序员可能会采用延迟初始化。但要正......
  • Java里面为什么搞了双重检查锁,写完这篇文章终于真相大白了
    双重检查锁定与延迟初始化在java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化。此时程序员可能会采用延迟初始化。但要......
  • Java Iterator(迭代器)的作用?
    JavaIterator(迭代器)不是一个集合,它是一种用于访问集合的方法,可用于迭代ArrayList和HashSet等集合。获取迭代器的方法:arrayList.Iterator()。参考1:https://www.runoo......
  • Java基础-浅拷贝和深拷贝
    浅拷贝浅拷贝会在堆上创建一个新的对象,如果原对象的属性是一个引用类型,拷贝的内部对象是原对象内部对象的引用地址,即原对象和拷贝对象用的是同一个内部对象。classInner......
  • JavaScript的this指向详解
    一、概念:函数的上下文(this)由调用函数的方式决定,function是“运行时上下文”策略;函数如果不调用,则不能确定函数的上下文。二、规则:对象打点调用它的方法函数,......
  • 【2023.02.01】在PVE上安装MacOS 13 Ventura
    【2023.02.01】在PVE上安装MacOS13Ventura本文参考链接:InstallingmacOS13VenturaonProxmox7.2–NicholasSherlock本次平台是i99980hk,CPU尽量新一点应该都可......