java 使用线程池进行多线程编程

本文介绍了一个使用Java实现的简单爬虫执行器,通过固定线程池的方式爬取指定数量的网页,并在爬取完成后自动关闭线程池。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

package com.gxk;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;

public class ExecutorTest {

    private static Integer pages=1; // 网页数

    private static boolean exeFlag=true; // 执行标识

    public static void main(String[] args) {
        ExecutorService executorService=Executors.newFixedThreadPool(10); // 创建ExecutorService 连接池默认连接10个 
        while(exeFlag){
            if(pages<=100){
                executorService.execute(new Runnable() {

                    @Override
                    public void run() {
                        // TODO Auto-generated method stub
                        System.out.println("爬取了第"+pages+"网页...");
                        pages++;
                    }
                });
            }else{
                if(((ThreadPoolExecutor)executorService).getActiveCount()==0){ // 活动线程个数是0
                    executorService.shutdown(); // 结束所有线程
                    exeFlag=false;
                    System.out.println("爬虫任务已经完成");
                }
            }

            try {
                Thread.sleep(100); // 线程休息0.1秒
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } 
        }

    }
}
### Java 线程池多线程编程示例教程 #### 创建并配置线程池 为了高效地利用资源,在Java中可以借助`java.util.concurrent.Executors`工厂方法来创建不同类型的线程池。常见的有固定大小的线程池(`newFixedThreadPool`)、缓存型线程池(`newCachedThreadPool`)以及计划任务执行服务(`ScheduledExecutorService`)等[^1]。 ```java import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; // 定义一个具有四个核心线程的工作队列 ExecutorService executor = Executors.newFixedThreadPool(4); ``` #### 提交任务给线程池 一旦有了线程池实例,就可以通过提交Runnable或Callable对象的方式让其异步执行特定的任务逻辑。对于只需要运行而无需返回结果的情况适用前者;如果希望获取计算后的成果,则应采用后者配合Future接口使用[^2]。 ```java public class Task implements Runnable { private final String name; public Task(String name) { this.name = name; } @Override public void run() { System.out.println(Thread.currentThread().getName() + ": Executing task " + name); try { Thread.sleep((long)(Math.random() * 1000)); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } System.out.println("Task " + name + " completed."); } } for(int i=0; i<10; ++i){ executor.submit(new Task("task-" + i)); } ``` #### 关闭线程池 当不再需要继续向线程池提交新任务时,应当调用shutdown()方法通知它停止接收新的请求,并等待已提交作业完成后再彻底关闭。这有助于防止内存泄漏和其他潜在问题的发生。 ```java executor.shutdown(); // 停止接受新任务, 已经分配出去的任务会继续执行直到结束. try{ if(!executor.awaitTermination(800, TimeUnit.MILLISECONDS)){ executor.shutdownNow(); // 如果超时则强制终止所有正在处理中的任务. } }catch(InterruptedException ex){ executor.shutdownNow(); } ``` #### 最佳实践建议 - **合理设置线程数量**:依据应用特点和硬件条件设定合适的并发度,过多过少都会影响效率。 - **异常处理机制**:确保每个工作单元内部都有完善的错误捕获流程,以免未被捕获的异常导致整个程序崩溃。 - **资源释放策略**:及时清理已完成工作的临时文件或其他占用空间较大的数据结构,保持系统稳定性和安全性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值