【Java并发】synchronized锁住的内容

synchronized用在方法上锁住的是什么?

锁住的是当前对象的当前方法,会使得其他线程访问该对象的synchronized方法或者代码块阻塞,但并不会阻塞非synchronized方法。

synchronized(this)锁住的是什么?

锁住的是当前的对象。当synchronized块里的内容执行完之后,释放当前对象的锁。同一时刻若有多个线程访问这个对象,则会被阻塞。

synchronized(object)锁住的什么?

锁住的是object对象。当synchronized块里的内容执行完之后,释放object对象的锁。同一时刻若有多个线程访问这个对象,则会被阻塞。

这里需要注意的是如果object为Integer、String等等包装类时(new出的对象除外),并不会锁住当前对象,也不会阻塞线程。因为包装类是final的,不可修改的,如果修改则会生成一个新的对象。所以,在一个线程对其进行修改后,其他线程在获取该对象的锁时,该对象已经不是原来的那个对象,所以获取到的是另一个对象的锁,所以不会产生阻塞。

例子:

public class Test7 {
    public static void main(String[] args) throws InterruptedException {
        Stu stu1 = new Stu(1);
        Thread t1 = new Thread(new MyThread(stu1));
        Thread t2 = new Thread(new MyThread(stu1));
        t1.start();
        t2.start();
    }
}

class MyThread implements Runnable{
    public MyThread(Stu stu){
        this.stu = stu;
    }
    private Stu stu;
    @Override
    public void run() {
        try {
            stu.a();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

class Stu{
    private Integer count ;
    public  Integer getCount(){
        return this.count;
    }
    public Stu(Integer count){
        this.count = count;
    }
    public  void a() throws InterruptedException {
        synchronized(count) {
            System.out.println(Thread.currentThread().getName()+"进入");
            count++;
            for (int i=0;i<3;i++){
                Thread.sleep(500);
                System.out.println(Thread.currentThread().getName()+"修改count");
            }
        }
    }
}

上边的代码中,开启了两个线程,分别对count进行++操作。虽然传入的都是实例stu1,两个线程中的count也是同一个对象,指向同一个实例。但是在一个线程对count进行++操作后,count不在是原来的count,而是一个新的Integer对象,这时第二个线程在访问count时,获取到的并不是原来的count的锁,而是新生成的Integer实例,这个实例并没有线程获取它的锁,因此第二个线程不会等待第一个线程执行完毕后再执行,而是交替执行。

输出结果如下:

当把

Stu stu1 = new Stu(1);

改为:

Stu stu1 = new Stu(new Integer(1));

时,输出如下:

因为new Integer(1)是一个对象,不会自动拆箱,所以,会按照先后顺序阻塞到第一个线程完毕后,第二个线程才开始执行。

posted @ 2018-04-13 17:04  SEC.VIP_网络安全服务  阅读(95)  评论(0编辑  收藏  举报