闲聊 “synchronized” 关键字

目录

Synchronized使线程安全

synchronized对象锁

synchronized 拥有锁重入的功能


Synchronized使线程安全

线程安全与非线程安全是使用多线程的时候都会遇到的经典的问题。我所理解的“非线程安全”也就是多个线程同时访问同一对象中的实例变量所产生一些脏读的现象。而线程安全就是以获得的实例变量的值都是经过同步处理的,从而不会产生脏读的现象。其中使结果线程安全的最简单的方法就是加上一个重量级的锁,用synchronized关键字进行修饰。


以下这种情况将会出现非线程安全的情况,代码如下

package com.huawei.bes.demo;

/**
 * Created by lenovo12 on 2018/11/5.
 */
public class SharedNum {
    private int num = 0;
    public  void addI(String username) throws InterruptedException {
        if("a".equals(username))
        {
            num = 100;
            System.out.println("a is over");
        }
        else
        {
            num = 200;
            System.out.println("b is over");
        }
        System.out.println(username + "num = " + num);
    }

    public static void main(String[] args) {
        SharedNum sharedNum = new SharedNum();
        ThreadA a = new ThreadA(sharedNum);
        ThreadB b = new ThreadB(sharedNum);
        a.start();
        b.start();
    }
}
class ThreadA extends  Thread{
    private SharedNum sharedNum ;
    public ThreadA(SharedNum sharedNum)
    {
        this.sharedNum = sharedNum ;
    }
    @Override
    public void run()
    {
        try {
            sharedNum.addI("a");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
class ThreadB extends  Thread{
    private SharedNum sharedNum ;
    public ThreadB(SharedNum sharedNum)
    {
        this.sharedNum = sharedNum ;
    }
    @Override
    public void run()
    {
        try {
            sharedNum.addI("b");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

运行结果如下,产生了脏读的现象

 此时若在SharedNum类中的addI方法进行synchronized修饰,则有结果发现是同步进行的,线程安全了。

synchronized对象锁

构建一个BESObject类,里面有两个方法methodA 与 methodB 其中只有methodA是同步的。然后启动两个线程执行分别执行这两个方法,代码如下。

package com.huawei.bes.demo;

/**
 * Created by lenovo12 on 2018/11/5.
 */
public class BESObject {
    public synchronized  void methondA() throws InterruptedException
    {
        System.out.println("begin methodA  Thread Name is " + Thread.currentThread().getName());
        Thread.sleep(1000);
        System.out.println("end method A " + System.currentTimeMillis());
    }
    public  void methondB() throws InterruptedException
    {
        System.out.println("begin methodB  Thread Name is " + Thread.currentThread().getName());
        Thread.sleep(1000);
        System.out.println("end method B " + System.currentTimeMillis());
    }

    public static void main(String[] args) {
        BESObject besObject = new BESObject();
        ThreadC c = new ThreadC(besObject);
        ThreadD d = new ThreadD(besObject);
        c.setName("C");
        d.setName("D");
        c.start();
        d.start();
    }
}
class ThreadC extends  Thread{
    private BESObject besObject;
    public ThreadC(BESObject besObject)
    {
        this.besObject = besObject;
    }
    @Override
    public void run()
    {
        try {
            besObject.methondA();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

class ThreadD extends  Thread{
    private BESObject besObject;
    public ThreadD(BESObject besObject)
    {
        this.besObject = besObject;
    }
    @Override
    public void run()
    {
        try {
            besObject.methondB();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

运行结果如下:

接着我们将methodB也使用“synchronized” 关键字进行修饰,运行结果如下

所以这个结果我们可以得出一个结论了:

  • 线程C先持有object对象的锁,D线程还是可以异步的方式调用object对象中的非synchronized方法
  • 线程C先持有object对象的锁,D线程如果在这个时候调用object对象中的synchronized方法就会同步了。

synchronized 拥有锁重入的功能

可重入锁的就是:自己可以再次获取自己的内部锁,比如有一个线程获得了某个对象锁,这个时候这个对象锁还没有释放,当它再次想要获取这个对象的锁的时候还是可以获取的,如果synchronized不可重入锁的话就会造成死锁。下面代码解释一切

package com.huawei.bes.demo;


class MyThread extends  Thread{
    @Override
    public void run()
    {
        SynTest service = new SynTest();
        service.service1();
    }
}
public class SynTest {
    public synchronized void service1(){
        System.out.println("服务1");
        service2();
    }
    public synchronized void service2(){
        System.out.println("服务2");
        service3();
    }
    public synchronized void service3(){
        System.out.println("服务3");
    }

    public static void main(String[] args) {
        MyThread t = new MyThread();
        t.start();
    }

}

运行结果如下,由此可以得出结论 “synchronized”拥有可重入锁

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、付费专栏及课程。

余额充值