线程的同步机制

1 线程安全问题的原因:由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程有参与进来,导致共享数据存在安全问题

2 解决方法:必须让一个线程操作共享数据完毕以后,其它线程才有机会参与共享数据的操作

3 java如何实现线程的安全,现成的同步机制

synchronized(同步监视器){  //需要被同步的代码块(操作共同数据的代码)}

同步监视器:由任何一个类的对象充当,哪个线程获取此监视器,就执行大括号里被同步的代码

1)同步代码块

class Window2 implements Runnable{
    private int ticket = 100;  //不用static,只创建一个对象
    public void run(){
        while(true){
            synchronized(this){ //注意一个线程进入同步代码块中,在执行的其中的代码过程中其他线程不能进入,所以称为同步锁。synchronized一定要放在共享数据前后
            if(ticket>0){  //ticket是共享数据,所以同步从这个地方开始,如果放在while前,那么线程1一直占用cpu直到结束,线程2,3就进不来了
                try {
                    Thread.currentThread().sleep(20);
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }  //只能加try-catch,不能throws,因为子类方法异常不能大于父类
                System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--);
            }
            else{
                break;
            }}
class Window extends Thread{
    static int ticket = 100;
    static Object obj = new Object();
    public void run(){
        while(true){
            synchronized (obj) {
                if (ticket > 0) {
                    try {
                        Thread.currentThread().sleep(20);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + ":"
                            + ticket--);
                } else {
                    break;
                }
            }

 

2)同步方法: 将操作共享数据的方法声明为synchronized,此方法称为同步方法,能够保证当其中一个线程执行此方法时,其他线程在外等待直到此线程执行完此方法。只能在实现方式中能使用,继承方式3个对象,有3个this,不能用同步方法

同步方法的锁:this,不用显式指明。

package lianxi1;
//方式二:实现方式创建多线程
class Window3 implements Runnable{
    private int ticket = 100;  //不用static,只创建一个对象
    public void run(){
        while(true){
           show();
        }
        }
    public synchronized void show(){ //不是显式调用this,因为只有一个对象,所以只有this就是同步监视器
        if(ticket>0){  //ticket是共享数据,所以从这个地方开始
            try {
                Thread.currentThread().sleep(20);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }  //只能加try-catch,不能throws,因为子类方法异常不能大于父类
            System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--);
        }
    }
}
public class TestThreadWindow3 {

    public static void main(String[] args) {
        Window3 w2 = new Window3();
        Thread t1 = new Thread(w2); //将Runnable实现类对象作为形参传人Thread构造器,执行start方法
        Thread t2 = new Thread(w2);
        Thread t3 = new Thread(w2);
        t1.setName("窗口1");
        t2.setName("窗口2");
        t3.setName("窗口3");
        t1.start();
        t2.start();
        t3.start();

    }

}
posted on 2014-12-12 22:45  追梦的小屁孩  阅读(133)  评论(0)    收藏  举报