ThreadLocal能不能保证原子性?

每一个ThreadLocal能够放一个线程级别的变量,可是它本身能够被多个线程共享使用,并且又能够达到线程安全的目的,且绝对线程安全。

  1. public class ThreadLocalTest02 {
  2. private static int age = 20;
  3. public static void main(String[] args) {
  4. ThreadLocal<User> userThreadLocal = new ThreadLocal<>();
  5. User user = new User("张三", age);
  6. userThreadLocal.set(user);
  7. // 循环次数少了可能效果不明显
  8. for(int i=0; i<1000; i++){
  9. new Thread(new Runnable() {
  10. @Override
  11. public void run() {
  12. Thread currentThread = Thread.currentThread();
  13. userThreadLocal.set(user); // 为当前线程设置user
  14. userThreadLocal.get().setAge(++age); // 修改年龄
  15. // 查看每个子线程中的user信息
  16. System.out.println(currentThread+" "+userThreadLocal.get());
  17. }
  18. }).start();
  19. }
  20. // 给子线程足够的时间去完成计算
  21. try {
  22. Thread.sleep(3000);
  23. } catch(InterruptedException e) {
  24. e.printStackTrace();
  25. }
  26. /*
  27. * 查看main线程中的user
  28. * 按道理,上面循环了1000次,age累加1000次,这里的年龄应该是1020
  29. * 实际上,多线程会破坏数据的原子性,这里的age计算值可能小于1020
  30. */
  31. System.out.println("主线程"+" "+userThreadLocal.get());
  32. }
  33. }

结果

image.png

结论

  1. 各线程共享user,在一个线程中修改user,另一个线程可以看到修改
  2. ThreadLocal不能保证数据的原子性