开启多线程下变量共享与私有问题

开启多线程下变量共享与私有问题

🌵ThreadLocal和Atomic是Java中用于多线程编程的两个重要工具

ThreadLocal是一个线程局部变量,它为每个线程提供了独立的变量副本,确保每个线程都可以访问自己的变量副本而不会影响其他线程的变量。在多线程环境下,使用ThreadLocal可以避免线程安全问题。

Atomic是一组原子操作类,提供了一些常见的原子操作,如原子更新整型、原子更新引用等。在多线程环境下,使用Atomic可以保证操作的原子性,避免出现数据竞争和线程安全问题

🎯下面是一个简单的示例ThreadLocal和Atomic在多线程环境下的用法:

import java.util.concurrent.atomic.AtomicInteger;

public class ThreadLocalAndAtomicExample {
    
    private static ThreadLocal<Integer> threadLocal = ThreadLocal.withInitial(() -> 0);
    private static AtomicInteger atomicInteger = new AtomicInteger(0);
    
    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                int localValue = threadLocal.get();
                localValue++;
                threadLocal.set(localValue);
                
                int atomicValue = atomicInteger.incrementAndGet();
                
                System.out.println("ThreadLocal value: " + threadLocal.get() + " Atomic value: " + atomicValue);
            }).start();
        }
    }
}

在上面的示例中,我们创建了一个ThreadLocal变量和一个AtomicInteger变量,然后启动了5个线程,每个线程对这两个变量进行操作。通过输出结果可以看到,ThreadLocal变量在每个线程中独立维护了自己的值,而AtomicInteger变量则保证了操作的原子性。

总结来说,在多线程编程中,ThreadLocal可以用于保证每个线程拥有独立的变量副本,避免线程安全问题;而Atomic可以用于保证操作的原子性,避免数据竞争和线程安全问题。在实际开发中,我们可以根据具体需求选择合适的工具来保证多线程程序的正确性和性能。

🎯下面是一个类似的示例,多线程使用EasyExcel处理导出数据,使用ExecutorService、CountDownLatch、ThreadLocal和Atomic在多线程环境下的用法:

   public void exportAllApply(HttpServletResponse response, ApplyScope applyScope){
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
    String fileName = "APPLY" + +"-"+Calendar.getInstance().getTimeInMillis()+".xlsx";
    List<ApplyExportVo> allApply = new ArrayList<>();
    try {
        ExcelWriter excelWriter = EasyExcel.write(fileName, ApplyExportVo.class).build();
        WriteSheet writeSheet = EasyExcel.writerSheet("sheet1").build();
        int threadCount = 5;
        //默认开启五个线程
        ExecutorService executorService = Executors.newFixedThreadPool(threadCount);
        CountDownLatch latch = new CountDownLatch(threadCount);
        ThreadLocal<ApplyScope> threadLocalApply = ThreadLocal.withInitial(ApplyScope::new);
        AtomicLong updatePage = new AtomicLong(1);
        for (int i = 0; i < threadCount; i++) {
            int finalI = i+1;
            executorService.execute(() -> {
                ApplyScope localApply = threadLocalApply.get();
                BeanUtils.copyProperties(applyScope,localApply);
                while (true) {
                    long currentPage = updatePage.getAndIncrement();
                    //更新分页信息,需同步
                    updatePage(localApply, currentPage);
                    ApplyGeneralVo applyGeneralVo = this.queryApplyView(localApply);
                    List<ApplyVo> applyList = applyGeneralVo.getApplyList();
                    log.info("线程threadCount:"+finalI+"当前页码"+currentPage+",数量"+applyList.size());
                    if(CollectionUtils.isEmpty(applyList)){
                        break;
                    }
                    List<ApplyExportVo> exportVos = BeanUtil.toBeanList(applyList, ApplyExportVo.class);
                    allApply.addAll(exportVos);
                }
                latch.countDown();
            });
        }
        executorService.shutdown();

        latch.await();
        Long startWrite = Calendar.getInstance().getTimeInMillis();
        Optional.ofNullable(allApply).ifPresent(data -> {
            excelWriter.write(data, writeSheet);
        });
        excelWriter.finish();
        log.info("----- writeExcelTime:" + (Calendar.getInstance().getTimeInMillis() - startWrite));
    } catch (Exception e) {
        e.printStackTrace();
    }
}


private synchronized long updatePage(ApplyScope scope, long current){
    PageScope page = scope.getPages();
    if(page == null){
        page = new PageScope();
        page.setSize(1000L);
        page.setCurrent(current);
        scope.setPages(page);
    } else {
        page.setCurrent(current);
    }
    return page.getCurrent();
}

🚩在上面的示例中,我们创建了一个固定大小的线程池ExecutorService,并使用CountDownLatch来等待所有线程执行完毕。每个线程在执行时会对ThreadLocal变量和AtomicInteger变量进行操作,并在操作完成后调用CountDownLatch的countDown()方法来通知主线程。

通过输出结果可以看到,每个线程中的ThreadLocal变量和AtomicInteger变量都能够正确地保持独立和原子性,同时CountDownLatch确保了所有线程执行完毕后主线程才会继续执行。

  • 7
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值