线程安全相关bug 及 System.out.println()的隐藏用法

一位同学在群里说碰到 奇怪bug,示例代码如下

@Slf4j
public class Interesting {
    volatile int a = 1;
    volatile int b = 1;

    public void add() {
        log.info("add start");
        for (int i = 0; i < 10000; i++) {
            a++;
            b++;
        }
        log.info("add done");
    }

    public void compare() {
        log.info("compare start");
        for (int i = 0; i < 10000; i++) {
            //a始终等于b吗?
            if (a < b) {
                log.info("a:{},b:{},{}", a, b, a > b);
                //最后的a>b应该始终是false吗?
            }
        }
        log.info("compare done");
    }
}

他起了两个线程来分别执行 add 和 compare 方法:

Interesting interesting = new Interesting();
new Thread(() -> interesting.add()).start();
new Thread(() -> interesting.compare()).start();

按道理,a 和 b 同样进行累加操作,应该始终相等,compare 中的第一次判断应该始终不 会成立,不会输出任何日志。但,执行代码后发现不但输出了日志,而且更诡异的是, compare 方法在判断 ab 也成立:

群里一位同学看到这个问题笑了,说:“是线程安全问题嘛。很 明显,你这是在操作两个字段 a 和 b,有线程安全问题,应该为 add 方法加上锁,确保 a 和 b 的 ++ 是原子性的,就不会错乱了。”随后,他为 add 方法加上了锁:

public synchronized void add()

为什么锁可以解决线程安全问题呢。因为只有一个线程可以拿到锁,所 以加锁后的代码中的资源操作是线程安全的。但是,这个案例中的 add 方法始终只有一个线程在操作,显然只为 add 方法加锁是没用的。

之所以出现这种错乱,是因为两个线程是交错执行 add 和 compare 方法中的业务逻辑, 而且这些业务逻辑不是原子性的:a++ 和 b++ 操作中可以穿插在 compare 方法的比较代 码中;更需要注意的是,a<b 这种操作在字节码层面是加载 a、加载 b 和比较三步, 代码虽然是一行但也不是原子性的(volatile 只能保证有序性和可见性,不能保证原子性)。

所以,正确的做法应该是,为 add 和 compare 都加上方法锁,确保 add 方法执行时, compare 无法读取 a 和 b

总结:加锁前要清楚锁和被保护的对象是不是一个层面的

小插曲:

变量 a、b 都使用了 volatile 关键字,能不能不使用?

答案:

必须加volatile,因为volatile保证了可见性。改完后会强制让工作内存失效。去主存拿。如果不加volatile的话那么在 for 语句里面添加输出语句也是OK的。因为println源码加锁了,sync会让当前线程的工作内存失效。必须加volatile或者使用AtomicBoolean/AtomicReference等也行,后者相比volatile除了确保可见性还提供了CAS方法保证原子性。

### 回答1: System.out.printlnJava中的一个输出语句,用于将数据打印到控制台。它的使用方法如下: 1. System.out.println("Hello World"); //输出字符串 2. System.out.println(10); //输出整数 3. System.out.println(3.14); //输出浮点数 4. System.out.println(true); //输出布尔值 5. System.out.println('a'); //输出字符 6. System.out.println("Hello" + "World"); //输出连接后的字符串 7. System.out.println("Hello" + 10); //输出连接后的字符串和整数 注意:System.out.println语句会自动在输出的内容后面添加一个换行符。 ### 回答2: System.out.printlnJava语言中的一个输出语句,用于将指定的数据输出到控制台。它的使用方法如下: 1. System.out.println括号内使用双引号引起来的字符串,例如: System.out.println("Hello World!"); 输出结果为:Hello World! 2. System.out.println括号内可以直接输出变量的值,例如: int num = 10; System.out.println(num); 输出结果为:10 3. System.out.println可以输出多个变量的值,使用加号连接,例如: int a = 5; int b = 3; System.out.println("a的值是:" + a + ", b的值是:" + b); 输出结果为:a的值是:5, b的值是:3 4. System.out.println也可以输出数学表达式的结果,例如: int c = 10; int d = 5; System.out.println("c + d的结果是:" + (c + d)); 输出结果为:c + d的结果是:15 5. System.out.println还可以输出布尔类型的值,例如: boolean flag = true; System.out.println("flag的值是:" + flag); 输出结果为:flag的值是:true 需要注意的是,System.out.println在输出完指定的内容后会自动换行。如果不想换行,可以使用System.out.print方法。另外,System.out.println的语句可以在程序的任何地方使用,帮助我们在调试过程中观察程序的执行结果。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值