Java多线程探究-线程同步

来源:互联网 发布:javac找不到java文件 编辑:程序博客网 时间:2024/06/06 06:48

什么是线程同步

线程同步是解决多个线程同时对同一数据进行操作,而导致的数据破坏
就像前面我写的卖票的例子,就出现了卖出-1的票,这样就出现了问题

线程同步的方式

1. synchronized关键字

1.1 同步方法

在方便声明上加入synchronized,这样,这个方法就是同步了,每次只有一个线程可以执行方法的代码,如果这个线程没有离开方法,那么
其他线程只能处于阻塞状态,这样就保证了一次只有一个线程在操作数据

class MyRunnable implements   Runnable{    private int ticket = 100;    @Override    public  void run(){            salseTicket();    }    public synchronized void salseTicket(){        while (true){            if(ticket > 0){                try {                    Thread.sleep(5);                    System.out.println("线程 "+Thread.currentThread().getName()+" sales "+ticket--);                } catch (InterruptedException e) {                    e.printStackTrace();                }            }else{                break;            }        }    }}public class ThreadDemo {    public static void main(String[] args) {        MyRunnable runnable = new MyRunnable();        Thread t1 = new Thread(runnable,"one");        Thread t2 = new Thread(runnable,"two");        Thread t3 = new Thread(runnable,"third");        Thread t4 = new Thread(runnable,"four");        t1.start();        t2.start();        t3.start();        t4.start();    }}

1.2. 同步代码块

同步代码块不同与同步方法,同步代码块只对某一部分代码进行加锁,这样会比整个方法加锁效率要高
在我们的卖票程序主要是最ticke–这个操作不是原子操作,我们必须保证每个线程把打印和ticke–这两个过程都执行,才允许其他的
线程进来继续执行
同步代码块需要一个锁,这个锁是对象锁,对于要同步的线程,必须拥有的锁是一致的,这样才能保证同步
比如,下面的例子中的对象锁是this,这个this是MyRunnable的引用,只有一份,每个线程都拥有,所以可以做到同步

class MyRunnable implements Runnable {    private int ticket = 100;    @Override    public void run() {        salseTicket();    }    public void salseTicket() {        while (true) {            synchronized (this) {//同步代码块                if (ticket > 0) {                    try {                        Thread.sleep(5);                        System.out.println("线程 " + Thread.currentThread().getName() + " sales " + ticket--);                    } catch (InterruptedException e) {                        e.printStackTrace();                    }                } else {                    break;                }            }        }    }}

试想下面的对象锁new Object(),这样尽管加了对象锁,但是个线程拥有的都是一个新的Object对象,那么这个锁机制就会失效,起不到同步的作用

synchronized (new Object()) {    if (ticket > 0) {        try {            Thread.sleep(5);            System.out.println("线程 " + Thread.currentThread().getName() + " sales " + ticket--);        } catch (InterruptedException e) {            e.printStackTrace();        }    } else {        break;    }}

类锁,除了对象锁意外,还有一种类锁,如果代码块所在的方法是静态方法,那么synchronized的锁应该是类,而不是this.众所周知,静态
方法是没有对象的this引用的,下面的程序就是一个类锁

public static  void salseTicket() {    while (true) {        synchronized (MyRunnable.class) {            if (ticket > 0) {                try {                    Thread.sleep(5);                    System.out.println("线程 " + Thread.currentThread().getName() + " sales " + ticket--);                } catch (InterruptedException e) {                    e.printStackTrace();                }            } else {                break;            }        }    }}
0 0
原创粉丝点击