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

目录

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");
    }
}
相关推荐
林夕071 天前
Qt QML与C++混合编程实战指南
java·开发语言·数据库
Kiling_07041 天前
Java Map集合详解与实战
java·开发语言·python·算法
Dicky-_-zhang1 天前
云原生数据库实战:TiDB与CockroachDB对比选型与落地实践
java·jvm
一条泥憨鱼1 天前
Stream流-从进阶到起飞
java·ide·后端·stream
Devin~Y1 天前
大厂Java面试实战:Spring Boot微服务、Redis缓存、Kafka消息队列与Spring AI RAG
java·spring boot·redis·kafka·mybatis·spring mvc·hikaricp
qingfeng154151 天前
企业微信定时群发实战:API 如何实现批量消息自动发送?
java·开发语言·python·自动化·企业微信
qingfeng154151 天前
企业微信 API 可以做什么?
java·开发语言·python·自动化·企业微信
梧桐和风1 天前
2026 年 Java 趋势:AI 浪潮下,Java 会过时吗?
java·开发语言·人工智能
Hesionberger1 天前
LeetCode105:前序中序构建二叉树(三解法)
java·数据结构·python·算法·leetcode·深度优先
嗨嗨的迷子1 天前
JDK 17 远程调试连不上 5005:从 attach timeout 到 JDWP 监听地址变更
java·开发语言