Mobility

Mobility

马上订阅 Mobility RSS 更新: http://lichuanyang.top/atom.xml

java多线程实现三个字母顺序输出

2017年1月25日 16:59

主要还是通过一个例子加深一下对java多线程里wait,notify的理解,因此写了一个例子,三个线程分别输出A,B,C三个字母,控制这三个线程的执行顺序,从而实现ABCABCABC..这样的输出。

这个问题主要还是需要设计一下锁的策略,这里只是提供了一种方式:

每个线程占用两把锁,分别代表自己(self)和前一个线程(prev), 三个线程的持有锁情况如下表所示:

线程号prev锁self锁
Aca
Bab
Cbc

A 首先启动,持有ac, 运行后先释放a, b可以执行。

线程run方法代码如下:

    <span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">run</span>() {        <span class="hljs-keyword">while</span> (<span class="hljs-keyword">true</span>) {            <span class="hljs-keyword">synchronized</span> (prev) {                <span class="hljs-keyword">synchronized</span> (self) {                    System.out.print(name);                    self.notify();                }                <span class="hljs-keyword">try</span> {                    <span class="hljs-comment">//如果想控制输出速度, 可以将sleep加在此处</span>                    <span class="hljs-comment">//如果加在sout之后,会导致c线程启动并占有b锁之后,a线程才会释放a锁,输出顺序会变成acbacb</span>                    <span class="hljs-comment">//也可以加大三个线程启动的间隔时间解决这一问题</span>                    <span class="hljs-keyword">try</span> {                        Thread.sleep(<span class="hljs-number">1000</span>);                    } <span class="hljs-keyword">catch</span> (InterruptedException e) {                        e.printStackTrace();                    }                    prev.wait();                } <span class="hljs-keyword">catch</span> (InterruptedException e) {                    e.printStackTrace();                }            }        }

需要注意的是,如果想控制输出速度,需要考虑一下sleep的位置和时间,避免在A线程执行完并释放a锁之前,C线程已经启动并持有了B锁,导致B线程无法正常启动。