两个线程同时对同个内存变量调用了i++ 100次,结束后i值可能的变化

有很多博主说是200,这没错,确实有可能是200,但是!

首先,i++的操作并不是单独的一条指令,他实际在内部是由三条指令构成的

1.首先,他把i的值从CPU寄存器中拿出来
2.然后,把寄存器中的值+1
3.最后,把寄存器写回到内存中

在多核CPU中:

他也需要执行:100 * 2 * 3 = 600条指令
首先结论为:最小是2,最大是200
首先为什么是2呢?因为存在下面的这种可能:
1.线程1执行第一次i++,把i从寄存器中取出来,这时i为0,i+1,这时寄存器CPU1中寄存器值为1,内存中为0
2.线程2执行第二次i++,把i从寄存器中取出来,这时i为0,i+1,这时寄存器CPU2中寄存器值为1,内存中为0
3.线程1继续执行第99次操作,并把值写到内存中,此时CPU1中寄存器的值为99,内存中为99
4.线程2继续执行第一次操作,将其值放回内存,此时CPU2中的寄存器值为1,内存中为1
5.线程1执行第100次i++,将内存中的值取回CPU1的寄存器,并执行加1,此时CPU1的寄存器中的值为2,内存中为1
6.线程2执行完所有操作,并将其放回内存,此时CPU2的寄存器值为100,内存中为100
7.线程1执行100次操作的最后一部分,将CPU1中的寄存器值放回内存,内存中值为2
至于为什么是200?
1.线程1执行完第一次i++,把结果写回内存,此时内存中值为1.
2.线程2抢到CPU,执行第一次i++,把结果写回内存,此时内存中值为2.
3.两个线程交替执行,最终结果为200

**

最重要的:

多核CPU有多个寄存器!!!

**

  • 6
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
这个问题可以通过使用Java的多线程编程来实现。首先,需要定义一个可共享的整型变量,即多个线程都可以对其进行修改。然后,创建两个加法线程两个减法线程,它们同时对该变量进行加法和减法操作,最后输出变量。 以下是一个可能的解决方案: ```java public class Main { public static void main(String[] args) { Counter counter = new Counter(); // 实例化计数器对象 AddThread addThread1 = new AddThread(counter); // 实例化加法线程对象 AddThread addThread2 = new AddThread(counter); SubThread subThread1 = new SubThread(counter); // 实例化减法线程对象 SubThread subThread2 = new SubThread(counter); addThread1.start(); // 启动线程 addThread2.start(); subThread1.start(); subThread2.start(); try { addThread1.join(); // 等待线程执行完毕 addThread2.join(); subThread1.join(); subThread2.join(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("变量为:" + counter.getCount()); // 输出变量 } } class Counter { private int count = 0; public synchronized void add() { // 加法操作 count++; } public synchronized void sub() { // 减法操作 count--; } public int getCount() { // 获取变量 return count; } } class AddThread extends Thread { private Counter counter; public AddThread(Counter counter) { this.counter = counter; } public void run() { for (int i = 0; i < 10000; i++) { counter.add(); // 进行加法操作 } } } class SubThread extends Thread { private Counter counter; public SubThread(Counter counter) { this.counter = counter; } public void run() { for (int i = 0; i < 10000; i++) { counter.sub(); // 进行减法操作 } } } ``` 在上述代码中,定义了一个`Counter`类,它包含一个可共享的整型变量`count`,并提供了加法和减法操作的方法。`AddThread`和`SubThread`类分别表示加法线程和减法线程,它们在运行时,通过调用`Counter`类的方法来修改变量。 需要注意的是,在`Counter`类的方法上加上`synchronized`关键字,可以保证多个线程访问该方法时的线程安全性。此外,使用`join()`方法可以等待线程执行完毕后再输出结果。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值