带你快速看完9.8分神作《Effective Java》—— 并发篇(工作里的这些坑你都遇到过吗?)

本文介绍了《Effective Java》中关于并发编程的部分,强调了同步访问共享可变数据的重要性,避免过度同步的策略,提倡使用Executor、Task和Stream替代线程,并推荐使用并发工具而非wait和notify。文章还讨论了并发集合和同步器的概念,提醒程序员在文档中明确线程安全性,并谨慎对待延迟初始化和依赖线程调度器的问题。
摘要由CSDN通过智能技术生成

🍊 Java学习:Java从入门到精通总结

🍊 Spring系列推荐:Spring源码解析

📆 最近更新:2022年1月13日

🍊 个人简介:通信工程本硕💪、阿里新晋猿同学🌕。我的故事充满机遇、挑战与翻盘,欢迎关注作者来共饮一杯鸡汤

🍊 点赞 👍 收藏 ⭐留言 📝 都是我最大的动力!

78 同步访问共享的可变数据

当多个线程共享可变数据的时候,每个读或者写数据的线程都必须执行同步


关键字 synchronized 可以保证在同一时刻,只有一个线程可以执行某一个方法,或者某一个代码块。

同步不仅可以阻止一个线程看到对象处于不一致的状态之中,它还可以保证进入同步方法或者同步代码块的每个线程,都能看到由同一个锁保护的之前所有的修改效果。

Java语言规范保证读或者写一个变量是原子的(atomic),除非这个变量的类型为long或者double。


虽然语言规范保证了线程在读取原子数据的时候,不会看到任意的数值,但是它并不保证一个线程写入的值对于另一个线程将是可⻅的。为了在线程之间进行可靠的通信,也为了互斥访问,同步是必要的

这归因于Java语言规范中的内存模型(JMM),它规定了一个线程所做的变化何时以及如何变成对其他线程可⻅


如果对共享的可变数据的访问不能同步,其后果将非常可怕,例如下面这段程序:

public class StopThread {
   
	private static Boolean stopRequested;
	
	public static void main(String[] args) throws InterruptedException {
   
		Thread backgroundThread = new Thread(() -> {
   
			int i = 0;
			while (!stopRequested)
				i++;
		});
		backgroundThread.start();
		TimeUnit.SECONDS.sleep(1);
		stopRequested = true;
	}
}

我们原本设想的是,这段程序运行大约一秒钟左右,之后主线程将stopRequested设置为true,使后台线程的循环终止。

但实际运行结果却是程序永远不会终止!问题在于,由于没有同步,就不能保证后台线程何时看到主线程对stopRequested的值所做的改变。没有同步,虚拟机将以下代码:

while (!stopRequested)
	i++;

转变成这样:

if (!stopRequested)
	while (true)
		i++;

编译器的优化初衷是好的,但这里却帮了倒忙!


修正这个问题的一种方式是同步访问stopRequested字段。这个程序会如预期般在大约一秒之内终止:

public class StopThread {
   
    private static Boolean stopRequested;

    private static synchronized void requestStop() {
   
        stopRequested = true;
    }

    private static synchronized Boolean stopRequested() {
   
        return stopRequested;
    }

    public static void main(String[] args)
            throws InterruptedException {
   
        Thread backgroundThread = new Thread(() -> {
   
            int i = 0;
            while (!stopRequested())
                i++;
        });
        backgroundThread.start();
        TimeUnit.SECONDS.sleep(1);
        requestStop();
    }
}

注意读和写操作都要被同步,否则无法保证同步能起作用


还是有其他更正确的替代方法,它更加简洁,性能也可能更好。虽然volatile修饰符不执行互斥访问,但它可以保证任何一个线程在读取该字段的时候都将看到最近刚刚被写入的值:

public class StopThread {
   
	private static volatile Boolean stopRequested;
	public static void main(String[] args) throws InterruptedException {
   
		Thread backgroundThread = new Thread(() -> {
   
			int i = 0;
			while (!stopRequested)
				i++;
		});
		backgroundThread.start();
		TimeUnit.SECONDS.sleep(1);
		stopRequested = true;
	}
}

在使用volatile的时候务必要小心。以下面的方法为例,假设它要产生序列号:

private static volatile int nextSerialNumber = 0;

public static int generateSerialNumber() {
   
	return nextSerialNumber++;
}

这段代码是有问题的,原因在于:

评论 144
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小王曾是少年

如果对你有帮助,欢迎支持我

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值