线程池与Threadlocal

线程池与Threadlocal

线程池: 线程池是为了使线程能够得到循环的利用,线程池里面养着一些线程,有任务需要使用线程的时候就往线程池里抓线程对象出来使用。线程池里的线程能够重复使用,所以在资源上能够得到比较好的利用。 在任务数量多的时候就适合使用线程池,因为总不可能将线程无限的开启下去,万一任务数量有几千的话就得开几千个线程,这样对于资源上就比较浪费了。如果使用线程池的话,就能重复的利用线池里的线程,就不需要一直新开启线程,所有的线程就能得到很好的循环利用。

线程池的开启方式:

线程池通过Executors类来开启,线程池有几种类型,有固定线程数量的,有不固定数量的,有具备定时功能的,还有单线程的。

1.创建固定线程数量的线程池:

image

运行结果:

image

2.创建不固定线程数量的线程池:

image

运行结果:

image

3.创建具备定时功能的线程池:

image

运行结果:

image

4.创建单线程池:

image

运行结果:

image

Threadlocal类:

说Threadlocal类之前,先看看一个问题。如果有A、B、C、D、E这几个方法,这几个方法除了A方法外都不具备参数,但是想要从A方法将一个值传递到E方法上去怎么办?仔细想一想似乎使用一个静态属性作为一个中间介质就可以实现到传递的效果。

示意图:

image

虽然咋看之下好像没什么问题,但是问题在于如果是多个线程同时去调用的话,就会出现值被覆盖的问题了,数据上就会出现混乱了。

代码示例: image image

运行结果:

image

从代码的运行结果可以看出,数据有被覆盖的现象。这时候可能会有人说,每次都构建Test1的对象来进行调用就可以避免出现这种问题了。是的,的确每次构建一个对象就能避免这种问题的出现,但是万一是静态的情况呢,不是说在开发的过程中都只会出现一种情况,所以如果是这种静态的情况,使用这个办法就没办法解决了,甚至还会出现问题。

虽然上面这种办法用在实例的情况下还行,不过还有另一种方法,就是使用Hashtable集合就能解决这个问题,Hashtable集合是键值对集合,一个键对应着一个值。因为这种特性,我们可以将线程的名称作为键值,然后对应存储这个线程所带的值,接着存放在Hashtable集合里。这样,在方法中取值的时候就能取到与线程相对应的值了,所以其他线程的值就不会被覆盖了,使用这个方法就能解决静态的情况了。

正题到了,使用Hashtable集合还是稍微有点麻烦,所以还有一种更方便的解决方式就是使用Threadlocal类,使用Threadlocal类的set将值设置进去,再使用get方法得到值就可以了,都不需要设置键值,比起Hashtable要方便一些。

代码示例:

image image

运行结果:

image

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
线程池使用ThreadLocal是为了在多线程环境下,每个线程都能够独立地访问自己的变量副本,而不受到其他线程的影响。ThreadLocal提供了一种线程局部变量的机制,使得每个线程都可以维护自己的变量副本。 在使用线程池时,我们可以通过在任务执行之前将需要共享的变量设置到ThreadLocal中,然后在任务执行期间获取该变量。这样就可以保证每个线程都能够独立地访问自己的变量副本。 下面是一个使用ThreadLocal的示例代码: ```java import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class ThreadPoolExample { private static ThreadLocal<Integer> threadLocal = new ThreadLocal<>(); public static void main(String[] args) { ExecutorService executorService = Executors.newFixedThreadPool(5); for (int i = 0; i < 10; i++) { final int index = i; executorService.execute(() -> { // 设置线程局部变量的值 threadLocal.set(index); System.out.println("Thread " + Thread.currentThread().getId() + ", value: " + threadLocal.get()); // 清除线程局部变量的值 threadLocal.remove(); }); } executorService.shutdown(); } } ``` 在上述代码中,我们创建了一个固定大小为5的线程池,并通过循环提交了10个任务。在每个任务中,我们将任务的索引存储到ThreadLocal中,并在任务执行期间获取该值并打印出来。最后,我们需要记得在任务执行结束后,清除ThreadLocal中的值,以防止内存泄漏。 通过使用ThreadLocal,每个线程都能够独立地访问自己的变量副本,从而避免了线程安全的问题,并且能够有效地利用线程池来处理多个任务。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值