龙空技术网

面试题:java用两个线程交替打印从0到100,很多人都做错

老嗨 7577

前言:

当前各位老铁们对“java奇数偶数分别打印”大体比较珍视,姐妹们都需要了解一些“java奇数偶数分别打印”的相关文章。那么小编也在网上汇集了一些关于“java奇数偶数分别打印””的相关资讯,希望朋友们能喜欢,同学们一起来了解一下吧!

java用两个线程交替打印从0到100,这道题考的是多线程间的通信问题,有很多种实现方式。

我们先来看一种比较普遍的错误写法,定义2个方法,一个打印偶数,一个打印奇数,代码如下:

乍一看好像没什么问题,但是多运行几次就会发现有时候能正确输出,有时候则不能。因为2个线程同时竞争锁的时候,这种情况下并不能保证打印偶数的线程首先能竞争到锁。

我们将上面的代码稍微修改下,加一个变量控制,代码如下:

这里又有人会问:这个j是共享变量,能保证可见性吗?根据JSR133给Java内存模型定义的happen-before规则,针对于同一锁,A线程修改的变量happen-befoe于B线程对同一变量的读。线程进入synchronized代码块前后,执行过程入如下:

1、线程获得互斥锁

2、清空工作内存

3、从主内存拷贝共享变量最新的值到工作内存中作为副本

4、执行代码

5、将修改后的副本的值刷新回主存中

6、线程释放锁

随后,其他代码在进入synchronized代码块的时候,所读取到的工作内存上共享变量的值都是上一个线程修改后的最新值。

我们再来看一种比较简单的写法,代码如下:

上面的代码,我们同时启动多个线程也能交替打印0-100,这种写法是比较灵活的。

我们还可以使用ReentrantLock和AtomicInteger实现。

小结:1、当线程执行wait()方法时候,会释放当前的锁,然后让出CPU,进入等待状态。

2、只有当 notify/notifyAll() 被执行时候,才会唤醒一个或多个正处于等待状态的线程,然后继续往下执行,直到执行完synchronized 代码块的代码或是中途遇到wait() ,再次释放锁。

注意:notify/notifyAll() 的执行只是唤醒沉睡的线程,而不会立即释放锁。

3、根据JSR133给Java内存模型定义的happen-before规则,使用了synchronized关键字来进行同步可保证对同一个锁不同线程间的变量可见性。

标签: #java奇数偶数分别打印