synchronized关键字理解

synchronized是对象锁,也就是多线程操作某一个对象时对该对象的所有同步资源进行锁定。

// class MyObject 
public class MyObject {

    synchronized public void methodA() {
        try{
            System.out.println("begin methodA threadName=" + Thread.currentThread().getName());
            Thread.sleep(5000);
            System.out.println("end methodA, endTime=" + System.currentTimeMillis());
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
    }

    public void methodB() {
        try{
            System.out.println("begin methodB threadName=" + Thread.currentThread().getName()
                    + " begin time=" + System.currentTimeMillis());
            Thread.sleep(5000);
            System.out.println("end methodB, endTime=" + System.currentTimeMillis());
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
    }
}
// class ThreadA
public class ThreadA extends Thread {
    private MyObject object;

    public ThreadA(MyObject object) {
        this.object = object;
    }

    @Override
    public void run() {
        super.run();
        object.methodA();
    }
}
// class ThreadB
public class ThreadB extends Thread {
    private MyObject object;

    public ThreadB(MyObject object) {
        this.object = object;
    }

    @Override
    public void run() {
        super.run();
        object.methodB();
    }

}
// main class
public static void main(String[] args) {
    myObject();
}

private static void myObject() {
    MyObject object = new MyObject();
    ThreadA threadA = new ThreadA(object);
    threadA.setName("A");
    ThreadB threadB = new ThreadB(object);
    threadB.setName("B");
    threadA.start();
    threadB.start();
}

输出结果:

begin methodA threadName=A
begin methodB threadName=B begin time=1496543751228
end methodA, endTime=1496543756228
end methodB, endTime=1496543756228

从上面的结果可以看出,线程A和线程B在调用MyObject 类的同一个对象的不同方法是异步的,线程A调用methodA是所属临界资源,需要加同步锁,但是线程B调用的methodB并不属于临界资源,即使线程A获得了对象的锁也不影响线程B调用非临界资源methodB。

我们将MyObject的methodB也加上synchronized关键字,将该方法修改为临界资源。如下所示:

public class MyObject {

    synchronized public void methodA() {
        try{
            System.out.println("begin methodA threadName=" + Thread.currentThread().getName());
            Thread.sleep(5000);
            System.out.println("end methodA, endTime=" + System.currentTimeMillis());
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
    }

    synchronized public void methodB() {
        try{
            System.out.println("begin methodB threadName=" + Thread.currentThread().getName()
                    + " begin time=" + System.currentTimeMillis());
            Thread.sleep(5000);
            System.out.println("end methodB, endTime=" + System.currentTimeMillis());
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
    }
}

其他代码不变,再运行一下结果如下:

begin methodA threadName=A
end methodA, endTime=1496543732497
begin methodB threadName=B begin time=1496543732497
end methodB, endTime=1496543737498

从以上输出结果可以看出,在线程A执行methodA时,线程B是在等待状态的,必须等线程A执行完methodA后释放了对象锁,然后线程B才可以执行methodB方法。

在线程A执行methodA时,线程A获得了MyObject类的对象object的锁,也就是object的所有被synchronized关键字修饰的变量和方法都只能被线程A所使用,其他线程必须等待线程A使用完成后释放对象锁后才能去获取object对象的锁,然后去执行object对象的方法。

结论:
1) A线程先持有object对象的lock锁,B线程可以以异步的方式调用object对象中的非synchronized类型的方法。

2) A线程先持有object对象的lock锁,B线程如果在这时调用object对象中的synchronized类型的方法则需要等待,也就是同步。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1. synchronized关键字在使用层面的理解 synchronized关键字是Java中用来实现线程同步的关键字,可以修饰方法和代码块。当线程访问被synchronized修饰的方法或代码块时,需要获取对象的锁,如果该锁已被其他线程获取,则该线程会进入阻塞状态,直到获取到锁为止。synchronized关键字可以保证同一时刻只有一个线程能够访问被锁定的方法或代码块,从而避免了多线程并发访问时的数据竞争和一致性问题。 2. synchronized关键字在字节码中的体现 在Java代码编译成字节码后,synchronized关键字会被编译成monitorenter和monitorexit指令来实现。monitorenter指令对应获取锁操作,monitorexit指令对应释放锁操作。 3. synchronized关键字在JVM中的实现 在JVM中,每个对象都有一个监视器(monitor),用来实现对象锁。当一个线程获取对象锁后,就进入了对象的监视器中,其他线程只能等待该线程释放锁后再去竞争锁。 synchronized关键字的实现涉及到对象头中的标志位,包括锁标志位和重量级锁标志位等。当一个线程获取锁后,锁标志位被设置为1,其他线程再去获取锁时,会进入自旋等待或者阻塞等待状态,直到锁标志位被设置为0,即锁被释放后才能获取锁。 4. synchronized关键字在硬件方面的实现 在硬件层面,锁的实现需要通过CPU指令和总线锁来实现。当一个线程获取锁时,CPU会向总线发送一个锁请求信号,其他CPU收到该信号后会进入自旋等待状态,直到锁被释放后才能获取锁。总线锁可以保证多个CPU之间的原子操作,从而保证锁的正确性和一致性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值