js中执行到一个if就停止的代码_Java 如何停止一个执行中的线程

引子

这个题也算面试常见题了,如果你回答 Thread.stop(),那么面试差不多要结束了......这里先给出答案

  1. 给线程发送中断
  2. 在自己的业务代码里,做到以下 2 件事
    • 捕获 InterruptedException 异常,退出线程的执行
    • 调用 isInterrupted 方法,若结果为 true,退出线程的执行

接下来详细说明,并延伸谈谈为什么要停止一个执行中的线程

阻塞

我们先要明白什么是阻塞,答案有很多,这里给出一个最简单的说法

凡是抛出 InterruptedException 异常的操作,都会导致阻塞,例如

Thread.sleep();Object.wait();Lock.lockInterruptibly();BlockingQueue.take();Future.get();

那么,问题来了:网络通信时,等待服务端返回数据是不是阻塞?

答案是:NO,因为这时抛出的异常是 IOException 而不是 InterruptedException

阻塞含义:抛出 InterruptedException 异常的操作

为什么我们要搞明白什么是阻塞呢?原因是 Thread.interrupt 就是我们退出阻塞的唯一方法(除非你的阻塞操作自带了超时检测)

如果一个执行中的线程正处于阻塞状态,要停止它,有且仅有一个办法:来一个 interrupt,然后捕获 InterruptedException 异常并退出

这也意味着要中断网络 IO,并没有那么 easy

非阻塞

如果是非阻塞的状态下,怎么停止线程呢?

这里没有银弹,首先还是来一个 interrupt,然后重要的是,在你的业务代码的任意节点(一般是关键节点),检查中断状态,例如下面的代码

if (Thread.currentThread().isInterrupted()) {    return;}

比如,我们刚才说过,网络通信时在等待服务器返回数据,如果这时你想停止,那么只能是在服务器返回数据后,检查中断状态,然后决定是处理服务器返回的数据,还是直接退出;当然,基于中断发送的时机,如果这时你还没有和远程服务器连接,那就可以直接退出了

或者,你的业务代码正在执行循环,可以在每次循环的开始处进行检查

如何发送中断

要停止一个执行中的线程,第一个步骤就是发送一个中断给线程,有以下几种方式

  • 如果你能拿到 thread 对象,调用其 interrupt 方法
  • 如果你拿到的是线程池返回的 Future,调用其 cancel 方法,请自行研究下该方法传递的参数
  • 当然,还有个玉石俱焚的方法:直接把线程池给 shutdown......请自行研究下 ExecutorService 的 shutdown 和 shutdownNow

为什么要停止执行中的线程

  • 首先,线程是一种资源,如果是从线程池里获取的,那么还是一种共享资源,提前停止执行,可以更早的归还到池里,增加整个线程池的吞吐率
  • 其次,需要被停止的操作,多半是无意义的操作,除了耗费 cpu 资源,占用更多的内存以外,毫无意义

举例说明

在广告实时竞价时,需要预估广告的转化率 CTR,这个 CTR 一般都是由一个专门的预估服务提供的,在我们的 dsp 实现中,每次竞价会向 2 个不同的 CTR 预估服务发起调用,这样某一个服务超时,可以采用另一个服务的预估数据

2 个预估服务是有优先级的,如果优先级更高的服务已经返回结果,那么另一个服务的返回结果就没有意义了,如果该服务尚未返回,我们就应该停止它

一次广告竞价,参与的广告往往成千上万,每个广告都要给出一个预估值,预估服务每次返回的数据量是极大的,能到几十上百 K,然后我们还要对该数据进行解析:比如先把字节流转为字符串,再用 json 格式解析字符串,得到一个有数千元素的列表

如果我们在预估服务返回数据,但还没有执行字节流转字符串时,就发送中断,那么后续生成字符串并解析的操作就无需进行,这样很明显就节约了内存

66e5f57941ee616bdfd09356c168ada3.png
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
由于您没有给出具体的技术要求和文件分片下载的实现方式,我将提供一种Java Web环境下使用HTTP协议实现大文件分片下载的前端和后端程序。 前端程序: 1. HTML页面 ```html <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>大文件分片下载</title> </head> <body> <h1>大文件分片下载</h1> <form id="download-form"> <label for="url-input">请输入文件下载链接:</label> <input id="url-input" type="text" name="url" required> <br><br> <label for="thread-input">请输入下载线程数:</label> <input id="thread-input" type="number" name="thread" min="1" value="1" required> <br><br> <button type="submit">开始下载</button> </form> <div id="progress-bar" style="width: 0%; height: 20px; background-color: #0f0;"></div> <div id="progress-text">0%</div> <br><br> <a id="download-link" style="display: none;">点击此处下载文件</a> <script src="download.js"></script> </body> </html> ``` 2. JavaScript代码 ```javascript const downloadForm = document.getElementById('download-form'); const progressBar = document.getElementById('progress-bar'); const progressText = document.getElementById('progress-text'); const downloadLink = document.getElementById('download-link'); downloadForm.addEventListener('submit', (event) => { event.preventDefault(); const url = downloadForm.url.value; const thread = downloadForm.thread.value; startDownload(url, thread); }); function startDownload(url, thread) { const xhr = new XMLHttpRequest(); xhr.open('POST', '/download'); xhr.setRequestHeader('Content-Type', 'application/json'); xhr.onreadystatechange = () => { if (xhr.readyState === XMLHttpRequest.DONE) { if (xhr.status === 200) { const response = JSON.parse(xhr.responseText); if (response.success) { downloadLink.href = response.downloadUrl; downloadLink.style.display = 'block'; } else { alert(response.message); } } else { alert('下载失败,请重试!'); } } }; xhr.upload.onprogress = (event) => { progressBar.style.width = `${event.loaded / event.total * 100}%`; progressText.innerHTML = `${event.loaded / event.total * 100}%`; }; xhr.send(JSON.stringify({url, thread})); } ``` 后端程序: 1. Maven依赖 ```xml <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies> ``` 2. Spring Boot配置 ```java @Configuration public class AppConfig { @Bean public ThreadPoolTaskExecutor taskExecutor() { ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); executor.setCorePoolSize(10); executor.setMaxPoolSize(10); executor.setQueueCapacity(100); executor.setThreadNamePrefix("download-thread-"); executor.initialize(); return executor; } } ``` 3. Controller代码 ```java @RestController public class DownloadController { @Autowired private ThreadPoolTaskExecutor taskExecutor; @PostMapping("/download") public ResponseEntity<?> download(@RequestBody DownloadRequest request) throws IOException { String downloadUrl = request.getUrl(); int threadCount = request.getThread(); if (downloadUrl == null || downloadUrl.isEmpty()) { return ResponseEntity.badRequest().body(new DownloadResponse(false, "下载链接不能为空!")); } if (threadCount <= 0) { return ResponseEntity.badRequest().body(new DownloadResponse(false, "下载线程数必须大于0!")); } String filename = downloadUrl.substring(downloadUrl.lastIndexOf('/') + 1); File file = new File(filename); if (file.exists()) { file.delete(); } HttpURLConnection connection = (HttpURLConnection) new URL(downloadUrl).openConnection(); connection.setRequestMethod("HEAD"); long fileSize = connection.getContentLengthLong(); connection.disconnect(); if (fileSize < 0) { return ResponseEntity.badRequest().body(new DownloadResponse(false, "获取文件大小失败!")); } long blockSize = fileSize / threadCount + 1; CountDownLatch latch = new CountDownLatch(threadCount); for (int i = 0; i < threadCount; i++) { long start = i * blockSize; long end = i == threadCount - 1 ? fileSize - 1 : (i + 1) * blockSize - 1; taskExecutor.execute(() -> { try { HttpURLConnection conn = (HttpURLConnection) new URL(downloadUrl).openConnection(); conn.setRequestMethod("GET"); conn.setRequestProperty("Range", "bytes=" + start + "-" + end); InputStream inputStream = conn.getInputStream(); RandomAccessFile output = new RandomAccessFile(file, "rw"); output.seek(start); byte[] buffer = new byte[1024]; int length; while ((length = inputStream.read(buffer)) > 0) { output.write(buffer, 0, length); } output.close(); inputStream.close(); } catch (IOException e) { e.printStackTrace(); } finally { latch.countDown(); } }); } try { latch.await(); } catch (InterruptedException e) { e.printStackTrace(); } return ResponseEntity.ok(new DownloadResponse(true, "下载成功!", filename)); } public static class DownloadRequest { private String url; private int thread; public String getUrl() { return url; } public void setUrl(String url) { this.url = url; } public int getThread() { return thread; } public void setThread(int thread) { this.thread = thread; } } public static class DownloadResponse { private boolean success; private String message; private String downloadUrl; public DownloadResponse(boolean success, String message) { this.success = success; this.message = message; } public DownloadResponse(boolean success, String message, String downloadUrl) { this.success = success; this.message = message; this.downloadUrl = downloadUrl; } public boolean isSuccess() { return success; } public void setSuccess(boolean success) { this.success = success; } public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } public String getDownloadUrl() { return downloadUrl; } public void setDownloadUrl(String downloadUrl) { this.downloadUrl = downloadUrl; } } } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值