Java多线程使用列子

需求:有这样一个需求,从数据库捞一批dsl,进行查询,看dsl是否正确,如果一个一个跑非常慢,使用Java多线程去跑,加速查询速度

思路:使用java线程池,设置好线程池的参数,创建好线程池,提交这些任务到线程池,返回值是一个Future,我们把所有Future放到一个List里,所有任务提交完,我们再遍历List<Future>,一个一个get结果即可

具体代码:

 1. 查询任务代码:

public class QueryTask implements Callable {

    private final String dsl;

    private final QueryService queryService;

    public QueryTask(QueryService queryService, String dsl) {
        this.dsl = dsl;
        this.queryService = queryService;
    }

    @Override
    public Object call() throws Exception {
        QueryResult queryResult = queryService.doQuery(dsl);
        log.info("queryResult is {}", JSON.toJSONString(queryResult));
        if (queryResult.getCode() != 200) {
           return dsl;
        }

        if(queryResult.getCode() ==200){
            List<DataResult> result = queryResult.getResult();
            for (DataResult dataResult : result) {
                if(dataResult.getResultStatus()!=SUCCESS){
                    return dsl;
                }
            }
        }
        return "";
    }
}

 2.导出接口代码:

@RequestMapping(value = "/export-multithreading", method = RequestMethod.GET)
    public void exportPhones(HttpServletResponse response) throws IOException{
        StopWatch stopWatch = new StopWatch();
        stopWatch.start();
        //1.创建一个线城池
        ExecutorService taskPool = new ThreadPoolExecutor(16,32,200L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue(5000),
                new ThreadFactoryBuilder().setNameFormat("thread-thirdPushMsgJob-runner-%d").build());
        
        //2.从数据获取一批dsl
        List<String> dslList = scriptService.batchBearDslToFinderDsl("event", new ArrayList<>());
        Integer removeNumber = 0;
        Integer totalNumber = dslList.size();
        //3.提交任务
        List<Future> futureList = new ArrayList<>();
        Iterator it = dslList.iterator();
        while(it.hasNext()){
            String dsl = (String) it.next();
            try {
                QueryTask aTask = new QueryTask(queryService,dsl);
                Future<String> submit = taskPool.submit(aTask);
                futureList.add(submit);
            }catch (Exception e){
                log.error("query failed {}",dsl);
                it.remove();
            }
        }
        //4.处理结果
        for (Future future : futureList) {
            try {
                String result = (String) future.get();
                if (!"".equals(result)) {
                    dslList.remove(result);
                    removeNumber++;
                }
            }catch (Exception e){
                log.error("future failed:{}",e.getMessage());
            }
        }
        log.info("dslList size:{},remove size:{}",dslList.size(),removeNumber);
        stopWatch.stop();
        dslList.add("useAble number:"+dslList.size()+"total number:"+totalNumber+"remove number:"+removeNumber);
        dslList.add(stopWatch.prettyPrint());
        //5.导出文件
        response.setContentType("text/plain");
        String fileName = System.currentTimeMillis()+"";
        response.setHeader("Content-Disposition", "attachment; filename=" + fileName + ".txt");
        BufferedOutputStream buff = null;
        StringBuilder write = new StringBuilder();
        String enter = "\r\n";
        ServletOutputStream outSTr = null;
        try {
            outSTr = response.getOutputStream(); // 建立
            buff = new BufferedOutputStream(outSTr);
            // 把内容写入文件
            if (dslList.size() > 0) {
                for (String s : dslList) {
                    write.append(s);
                    write.append(enter);
                    write.append(enter);
                }
            }
            buff.write(write.toString().getBytes(StandardCharsets.UTF_8));
            buff.flush();
            buff.close();
        } catch (Exception e) {
            log.error("export txt failed:",e);
        } finally {
            assert buff != null;
            buff.close();
            outSTr.close();
        }
    }

我这是以批量提交dsl进行查询为列,其他多线程操作类似

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
当然可以!一个常见的多线程例是计算圆周率的问题。我们可以使用线程来加速计算过程。 在这个例中,我们可以将圆周率的计算任务分成多个任务,并由不同的线程并行执行。每个线程负责计算一部分的圆周率,并将结果合并起来得到最终的结果。 以下是一个简单的多线程计算圆周率的例: ```python import threading # 全局变量,用于存储每个线程的计算结果 result = 0 # 计算圆周率的函数 def calculate_pi(start, end): global result partial_result = 0 # 计算部分圆周率 for i in range(start, end): partial_result += 1 / (i * 4 + 1) - 1 / (i * 4 + 3) # 将计算结果加入全局变量 result += partial_result # 创建多个线程并行执行计算任务 def main(): num_threads = 4 # 设置线程数量 threads = [] # 创建并启动线程 for i in range(num_threads): start = i * 1000000 # 每个线程计算的起始位置 end = (i + 1) * 1000000 # 每个线程计算的结束位置 thread = threading.Thread(target=calculate_pi, args=(start, end)) threads.append(thread) thread.start() # 等待所有线程执行完毕 for thread in threads: thread.join() # 打印最终结果 print("计算得到的圆周率为:", result * 4) if __name__ == "__main__": main() ``` 在这个例中,我们创建了4个线程来并行计算圆周率的部分值,然后将它们的结果累加得到最终的结果。每个线程计算的起始位置和结束位置通过参数传递给`calculate_pi`函数。 通过多线程的方式,我们可以加速圆周率的计算过程,提高程序的执行效率。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值