并发编程之【优雅地结束线程的执行】

目录

Thread类中的stop方法【不推荐】

查看源码

测试

不推荐使用的原因

使用标志位的判断来结束线程的执行【不推荐】

实现案例

不推荐使用的原因

通过对线程对象的封装来停止线程的执行【推荐】

实现思路

具体实现

测试

案例一:任务未超时

案例二:任务执行超时


Thread类中的stop方法【不推荐】

查看源码

通过源码可以看到stop()方法会强制线程停止运行

java 复制代码
package java.lang;

// ...

public
class Thread implements Runnable {
    // ...
    
    // Forces the thread to stop executing.
    @Deprecated
    public final void stop() {
        // ...

        stop0(new ThreadDeath());
    }

    // ...
}

测试

java 复制代码
package concurrency;

public class ThreadStopTest {
    public static void main(String[] args) {
        Thread t = new Thread(() -> {
            while (true) {

            }
        });
        t.start();

        // 1秒后查看t线程是否为活跃状态
        sleepTime(1_000);
        System.out.println("t线程是否活跃:" + t.isAlive());

        // 3秒后强制t线程停止运行
        long beginTime = System.currentTimeMillis();
        sleepTime(3_000);
        t.stop();
        while (true) {
            if (!t.isAlive()) {
                long endTime = System.currentTimeMillis();
                System.out.println((endTime - beginTime) + "ms");
                break;
            }
        }
    }

    public static void sleepTime(long time) {
        if (time <= 0) return;
        try {
            Thread.sleep(time);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }
}

不推荐使用的原因

  • 方法已废弃,不建议使用
  • 方法本身不安全

使用标志位的判断来结束线程的执行【不推荐】

实现案例

java 复制代码
package concurrency;

public class FlagTest {
    private volatile static boolean flag = true;

    public static void main(String[] args) {
        new Thread(() -> {
            while (flag) {

            }
        }).start();

        // 3秒后,通过改变flag标识来停止线程t
        try {
            Thread.sleep(3_000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        flag = false;
    }
}

不推荐使用的原因

  • 在很多场景下并没有机会去判断标志位(如任务长时间阻塞等待)

通过对线程对象的封装来停止线程的执行【推荐】

实现思路

  • 封装一个类,提供两个方法,分别为执行任务的方法和超时停止任务执行的方法
  • 通过daemon和interrupt来实现

具体实现

java 复制代码
package concurrency;

public class ThreadService {
    private Thread executeThread;

    private volatile boolean isFinish = false;

    /**
     * 执行任务
     */
    public void execute(Runnable task) {
        if (task == null) return;
        executeThread = new Thread(() -> {
            Thread t = new Thread(task);
            // 设置为守护线程
            t.setDaemon(true);
            t.start();
            try {
                // task执行完
                t.join();
                isFinish = true;
            } catch (InterruptedException e) {
                System.out.println("任务执行超时...");
            }
        });
        executeThread.start();
    }

    /**
     * 超时停止任务的执行
     */
    public void shutdown(long timeout) {
        if (timeout <= 0) return;
        long currentTime = System.currentTimeMillis();
        while (!isFinish) {
            // 超时,打断执行任务的线程,使其结束
            if (System.currentTimeMillis() - currentTime > timeout) {
                executeThread.interrupt();
                break;
            }

            // 减少空转次数
            try {
                Thread.sleep(1);
            } catch (InterruptedException e) {
            }
        }
    }
}

测试

案例一:任务未超时

java 复制代码
package concurrency;

public class ThreadServiceTest {
    public static void main(String[] args) {
        // 任务:从数据库中查询数据
        Runnable queryDataFromDB = () -> {
            try {
                // 这里该任务执行耗时10s
                Thread.sleep(10_000);
                System.out.println("queryDataFromDB task execute success!");
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        };

        // 执行任务
        ThreadService threadService = new ThreadService();
        long beginTime = System.currentTimeMillis();
        threadService.execute(queryDataFromDB);
        // 最大忍耐限度为20s,任务没有超过20s就按照实际耗时执行成功
        threadService.shutdown(20_000);
        long endTime = System.currentTimeMillis();
        System.out.println("总耗时:" + (endTime - beginTime) + "ms");
    }
}

案例二:任务执行超时

java 复制代码
package concurrency;

public class ThreadServiceTest {
    public static void main(String[] args) {
        // 任务:从数据库中查询数据
        Runnable queryDataFromDB = () -> {
            try {
                // 这里该任务执行耗时30s
                Thread.sleep(30_000);
                System.out.println("queryDataFromDB task execute success!");
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        };

        // 执行任务
        ThreadService threadService = new ThreadService();
        long beginTime = System.currentTimeMillis();
        threadService.execute(queryDataFromDB);
        // 最大忍耐限度为20s,如果任务执行超过20s则认为任务失败,停止任务
        threadService.shutdown(20_000);
        long endTime = System.currentTimeMillis();
        System.out.println("总耗时:" + (endTime - beginTime) + "ms");
    }
}
相关推荐
雨中飘荡的记忆1 小时前
ElasticJob分布式调度从入门到实战
java·后端
考虑考虑10 小时前
JDK25模块导入声明
java·后端·java ee
_小马快跑_11 小时前
Java 的 8 大基本数据类型:为何是不可或缺的设计?
java
Re_zero14 小时前
线上日志被清空?这段仅10行的 IO 代码里竟然藏着3个毒瘤
java·后端
洋洋技术笔记14 小时前
Spring Boot条件注解详解
java·spring boot
程序员清风1 天前
程序员兼职必看:靠谱软件外包平台挑选指南与避坑清单!
java·后端·面试
皮皮林5511 天前
利用闲置 Mac 从零部署 OpenClaw 教程 !
java
华仔啊2 天前
挖到了 1 个 Java 小特性:var,用完就回不去了
java·后端
SimonKing2 天前
SpringBoot整合秘笈:让Mybatis用上Calcite,实现统一SQL查询
java·后端·程序员
日月云棠2 天前
各版本JDK对比:JDK 25 特性详解
java