Java 线程 6 - volatile和ThreadLocal

13 篇文章 0 订阅
12 篇文章 2 订阅

参考:

Java 线程 0 - 前言


主要内容:

  1. volatile
  2. ThreadLocal

volatile

参考:

Java关键字volatile的理解与正确使用

Java并发编程:volatile关键字解析

volatile 关键字用于定义变量,保证变量的 可见性指令重排序

在现代处理器系统中,线程可能会从高速缓存中读取共享变量值,这样有可能会出现高速缓存中的变量值和内存中的变量值不一致

当一个共享变量被 volatile 修饰时,更新后的值会立即保存到内存中,而线程读取该共享变量时,也必须从内存读取

由于 volatile 关键字不能保证 有序性其适合修饰的共享变量应该仅包含原子操作,比如对 boolean 变量的操作。相对的,如果需要对变量进行加减操作,则不具备原子性


ThreadLocal

参考:

java.lang Class ThreadLocal

ThreadLocal 类可以为线程创建一个共享变量副本,每个线程对该变量的操作均不会影响其它线程

《疯狂Java讲义 16.10 线程相关类》中给出一个示例

定义一个账户类,该类仅包含线程名:

public class Account {

    private ThreadLocal<String> name = new ThreadLocal<String>() {
        @Override
        protected String initialValue() {
//            return super.initialValue();
            return "Hi zj";
        }
    };

    public Account(String name) {
        System.out.println("初始化前:" + this.name.get());
        setName(name);
        System.out.println("初始化后:" + this.name.get());
    }

    public String getName() {
        return name.get();
    }

    public void setName(String name) {
        this.name.set(name);
    }
}

可以在定义 ThreadLocal 对象时调用函数 initialValue 初始化(未初始化的对象值默认为空

定义一个线程类 MyThread,使用账户类作为成员变量,并保存其线程名:

public class MyThread extends Thread {

    private Account account;

    public MyThread(Account account, String name) {
        super(name);
        this.account = account;
    }

    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            if (i == 6) {
                account.setName(getName());
            }

            System.out.println(account.getName() + " i = " + i);
        }
    }
}

测试函数如下:

public static void main(String[] args) {
    Account account = new Account("Hello World");

    new MyThread(account, "thread-1").start();
    new MyThread(account, "thread-2").start();

    for (int i = 0; i < 10; i++) {
        if (i == 6) {
            account.setName(Thread.currentThread().getName());
        }

        System.out.println(account.getName() + " i = " + i);
    }
}

测试函数中共有 3 个线程:main / thread-1 / thread-2,由结果可知,ThreadLocal 对象各自为这 3 个线程保存了一个副本,它们的修改不会影响对方

其余参考:

为什么说”JAVA中的ThreadLocal是最简单的非阻塞同步”?

将ThreadLocal变量设置为private static的好处是啥?

ThreadLocal和synchronized的区别?

Java中的ThreadLocal通常是在什么情况下使用的?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值