Java线程的同步(协调)synchronized

【格式】

同步代码块:synchronized(Object){...} 关键字在代码块前,每次只允许一个线程调用此代码块。

Object为任何对象(一般用this),每个对象都有一个标志位(0锁住状态,1锁打开状态)。

同步方法:synchronized void method(){...} 关键字在方法前,每次只允许一个线程调用此方法

【使用场景】

多线程访问同一资源时,为防止产生冲突,使用synchronized对该资源(方法、代码块)加锁。

理解为:多人上同一个厕所,synchronized是门上的锁(0闭、1开)。一个线程运行到synchronized处,要观察锁的状态,如果闭(里面有人在使用),在门口继续等待。当锁为开(里面的运行完毕),则此线程进入使用,并锁上。

【Runnable接口】

使用Runnable接口管理多线程:如果不使用同一个接口,则每个线程对象new各自的接口,那么会产生多个厕所(下列中的num数据池),每人一个。脱离了多线程访问同一资源的范畴。

【条件语句】

if条件尽量前置,否则,先执行再判断。

 

【举例】:通过不同方式购买车票后,查询余票数目。3个线程:手机电脑端、柜台、自助售票机

class MyThreadRunnalbe implements Runnable {
    int num = 10;//车票总数

    public void run() {//线程要执行的任务
        while (true) {
            synchronized (this) {//同步(协调)代码块,加锁
                if (num > 0) {//条件尽量前置
                    try {
                        Thread.sleep(500);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + "购买,余票:" + num--);
                } else break;
            }
        }
    }
}

public class Demo {
    public static void main(String[] args) {
        MyThreadRunnalbe mtr = new MyThreadRunnalbe();//创建端口对象(以下3个线程共同使用这1个端口)
        Thread tA = new Thread(mtr, "手机电脑端");// 以该类对象分别实例化4个线程
        Thread tB = new Thread(mtr, "柜台");
        Thread tC = new Thread(mtr, "自助售票机");
//        //不可以分别new MyThreadRunnalbe(),3个接口使得3个线程各自执行,无法达到有序的目的
//        Thread tA = new Thread(new MyThreadRunnalbe(), "手机电脑端");
//        Thread tB = new Thread(new MyThreadRunnalbe(), "柜台");
//        Thread tC = new Thread(new MyThreadRunnalbe(), "自助售票机");
        tA.start();// 分别启动线程
        tB.start();
        tC.start();
    }
}

 或者,同步(协调)方法sell()

class MyThreadRunnalbe implements Runnable {
    int num = 10;//车票总数

    public void run() {//线程要执行的任务
        while (true) {
            sell();//调用sell方法
            if (num <= 0) break;
        }
    }

    public synchronized void sell() {//创建sell方法,并同步(协调)

        if (num > 0) {//条件尽量前置
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "购买,余票:" + num--);
        }
    }
}

public class Demo {
    public static void main(String[] args) {
        MyThreadRunnalbe mtr = new MyThreadRunnalbe();//创建端口对象(以下3个线程共同使用这1个端口)
        Thread tA = new Thread(mtr, "手机电脑端");// 以该类对象分别实例化4个线程
        Thread tB = new Thread(mtr, "柜台");
        Thread tC = new Thread(mtr, "自助售票机");
//        //不可以分别new MyThreadRunnalbe(),3个接口使得3个线程各自执行,无法达到有序的目的
//        Thread tA = new Thread(new MyThreadRunnalbe(), "手机电脑端");
//        Thread tB = new Thread(new MyThreadRunnalbe(), "柜台");
//        Thread tC = new Thread(new MyThreadRunnalbe(), "自助售票机");
        tA.start();// 分别启动线程
        tB.start();
        tC.start();
    }
}

 

posted @ 2018-09-02 15:35  夕西行  阅读(268)  评论(0编辑  收藏  举报