金山的一道有关线程的笔试题

来源:互联网 发布:淘宝网雪地靴 编辑:程序博客网 时间:2024/06/06 03:51

金山的一道有关线程的笔试题

在北京时参加金山校招的一道笔试题:

四个线程,a b c d ,共享一个变量 i, ab 为加线程,cd 为减线程,四个线程依次执行,顺序为 abcd,输出为 0 1 2 1 0 1 2 1 0……

当时觉得这道题还可以,不一会儿就写出来了,我的思路是:四个线程,ab为一个锁,cd为一个锁,两个锁交替执行,每个锁的两个线程交替执行。后来回校后,zw问这道题,我又重新打了一下,写了好长时间,老是不对,线程的执行顺序总是控制不好,线程语法也老出错,一阵o(╯□╰)o,基础太薄弱……搜了搜方茅塞顿开。

public class Jinsan2 {     static int i = 0;    static Object lock = new Object();    static boolean flag1 = true, flag2 = true;     public static void main(String args[]) {     Thread a = new Thread("a") {        @Override        public void run() {        for (;;) {            synchronized (lock) {            while (!flag1 || !flag2) {        // 可以过时,应 while                 try {                lock.wait();                } catch (InterruptedException e) {                }            }            System.out.println(Thread.currentThread().getName()+" "+i++ + " ");            flag2 = !flag2;            lock.notifyAll();            }        }        }    };    Thread b = new Thread("b") {        @Override        public void run() {        for (;;) {            synchronized (lock) {            while (!flag1 || flag2) {                 try {                lock.wait();                } catch (InterruptedException e) {                }            }            System.out.println(Thread.currentThread().getName()+" "+i++ + " ");            flag1 = false;            flag2 = !flag2;            lock.notifyAll();            }        }        }    };    Thread c = new Thread("c") {        @Override        public void run() {        for (;;) {            synchronized (lock) {            while (flag1 || !flag2) {                 try {                lock.wait();                } catch (InterruptedException e) {                }            }            System.out.println(Thread.currentThread().getName()+" "+i-- + " ");            flag2 = !flag2;            lock.notifyAll();            }        }        }    };    Thread d = new Thread("d") {        @Override        public void run() {        for (;;) {            synchronized (lock) {            while (flag1 || flag2) {                 try {                lock.wait();                } catch (InterruptedException e) {                }            }            System.out.println(Thread.currentThread().getName()+" "+i-- + " ");            flag1 = true;            flag2 = !flag2;            lock.notifyAll();            }        }        }    };    a.start();    b.start();    c.start();    d.start();    }}

还有一个更好的方法就是为每个线程都标一个ID,a b c d 分别对象 0 1 2 3 ,这样,所有线程都共享一个 count 变量,每个线程执行时,都 首先获count%4 的值,如果对应自己线程的ID,那么就执行,否则等待,id=0 || id=1 表示是 a b 线程,则对 i 进行 加操作,否则进行 减操作这样四个线程都可以用同一个逻辑进行操作

public class Test {    private static int count;    static int i;    private static Object lock = new Object();    static class Threads implements Runnable {int id;public Threads(int id) {    this.id = id;}@Overridepublic void run() {    for (;;) {synchronized (lock) {    while (id != count % 4) {try {    lock.wait();} catch (InterruptedException e) {}    }    if (id == 0 || id == 1) {System.out.println(Thread.currentThread().getName()+ " " + i++ + " ");    }    if (id == 2 || id == 3)System.out.println(Thread.currentThread().getName()+ " " + i-- + " ");    count++;    lock.notifyAll();}    }}    }    public static void main(String args[]) {Thread a = new Thread(new Threads(0), "a");Thread b = new Thread(new Threads(1), "b");Thread c = new Thread(new Threads(2), "c");Thread d = new Thread(new Threads(3), "d");a.start();b.start();c.start();d.start();    }}

另外我调用wait(),notify()时调用的是this的,正确的调用应该是哪个锁对象就调用哪个对象的方法,我这可怕的基本功(ˇˍˇ)