synchronized锁(方法锁, 代码块锁)

3 篇文章 0 订阅
1 篇文章 0 订阅

synchronized锁可以解决线程安全问题,但是相应的,只要是锁,就会带来性能开销,所以尽可能减小锁的范围尤为重要。

synchronized锁无非修饰普通方法,修饰静态方法,修饰代码块,我认为无非就两种(对象锁、类锁),只不过是锁的使用对象不同而已,实际上synchronized锁的作用范围,取决于使用对象的生命周期。接下来简单介绍几种不同影响范围的锁。

synchronized修饰普通方法(对象锁)

普通方法作用范围是对象实例,不可跨对象,所以多个线程不同对象实例访问此方法,互不影响,无法产生互斥。

public class SynchronizedDemo {
    // 修饰普通方法(实例方法)
    public synchronized void instanceMethod() {
        // TODO 业务逻辑
    }

    public static void main(String[] args) {
        SynchronizedDemo obj1 = new SynchronizedDemo();
        SynchronizedDemo obj2 = new SynchronizedDemo();
        new Thread(() ->{
            obj1.instanceMethod(); //多线程访问加锁普通实例方法,互不影响
        }).start();
        new Thread(() ->{
            obj2.instanceMethod();
        }).start();
    }
}

synchronized修饰静态方法(类锁)

静态方法是通过类访问,是类级别的跨对象的,所以锁的范围是针对类,多个线程访问互斥。

public class SynchronizedDemo {
    // 修饰静态方法(类方法)
    public synchronized static void staticMethod() {
        // TODO 业务逻辑
    }

    public static void main(String[] args) {
        new Thread(() ->{
            SynchronizedDemo.staticMethod(); //多线程访问加锁静态方法,互斥
        }).start();
        new Thread(() ->{
            SynchronizedDemo.staticMethod();
        }).start();
    }
}

synchronized修饰代码块

括号里是对象(类似对象锁)

作用范围是对象实例,不可跨对象,所以多个线程不同对象实例访问此方法,互不影响,无法产生互斥。

public class SynchronizedDemo {
    // 代码块锁(对象实例):锁的应用对象是当前对象实例,可以称之为对象锁
    public void method1() {
        synchronized (this) {
            // TODO 业务逻辑
        }
    }

    public static void main(String[] args) {
        SynchronizedDemo obj1 = new SynchronizedDemo();
        SynchronizedDemo obj2 = new SynchronizedDemo();
        new Thread(() ->{
            obj1.method1(); //代码块锁,后面是对象,多线程访问互不影响
        }).start();
        new Thread(() ->{
            obj2.method1();
        }).start();
    }
}

括号里是类(类似类锁)

虽然是通过对象访问的此方法,但是加锁的代码块是类级别的跨对象的,所以锁的范围是针对类,多个线程访问互斥。

public class SynchronizedDemo {
    // 代码块锁(类):锁的应用对象是User类,可以称之为类锁
    public void method2() {
        synchronized (User.class) {
            // TODO 业务逻辑
        }
    }

    public static void main(String[] args) {
        SynchronizedDemo obj1 = new SynchronizedDemo();
        SynchronizedDemo obj2 = new SynchronizedDemo();
        new Thread(() ->{
            obj1.method2(); //代码块锁,后面是类,多线程访问互斥
        }).start();
        new Thread(() ->{
            obj2.method2();
        }).start();
}

MarkWord对象头

对象头中会保存当前对象的锁标记,如果是lock的是对象实例,那么两个对象各自会保存锁标记,不是同一个锁标记,所以无法形成互斥。

查看对象头信息工具

加入pom依赖

<dependency>
    <groupId>org.openjdk.jol</groupId>
    <artifactId>jol-core</artifactId>
    <version>0.9</version>
</dependency>

使用方法

public class ClassLayoutDemo {
    public static void main(String[] args) {
        ClassLayoutDemo demo = new ClassLayoutDemo();
        // 打印对象头内存布局
        System.out.println(ClassLayout.parseInstance(demo).toPrintable());
    }
}

打印结果

前八个字节对应的是对象头信息,接下面4个字节是类元指针数据,最后4个字节是对齐填充。

synchronized锁的优化升级

JDK1.6以前只有无锁重量级锁两种,但是重量级锁太沉重,太消耗CPU性能;所以JDK1.6之后引入了偏向锁轻量级锁,对锁进行了优化升级,这是一个无锁化的实现,减轻线程阻塞和唤醒带来的CPU消耗。

无锁:没有加锁

偏向锁:没有其他线程竞争的情况下,线程A抢占锁,这个时候会偏向于线程A。偏向锁默认延迟开启。

轻量级锁:如果线程A抢占到锁,此时线程B来竞争,抢占锁失败,那么此时通过轻量级锁来避免线程被阻塞,一直在循环尝试获取锁,可以避免线程阻塞唤醒导致的用户态和内核态之间的切换,可以减轻CPU压力;如果循环一定次数后依然没有抢占到锁,那么才会升级为重量级锁。

自旋锁:自旋锁是轻量级锁的一种实现,通过for循环来自我循环去尝试获取锁,通过CAS机制去判断锁的标识状态。

重量级锁:有锁,线程会阻塞,直到抢占到锁被唤醒,会涉及到用户态和内核态之间的切换。

Lock锁与synchronized锁的区别

1) Lock锁需要手工释放,而synchronized锁会自动释放;
2) Lock锁有很多实现,有独占锁互斥锁重入锁,还有共享锁,而synchronized是互斥锁重入锁;
3) Lock是个接口,有很多锁的实现类,而synchronized只是个关键字;
4) Lock只能作用在代码块上,而synchronized既可以修饰代码块还是修饰方法;
5) Lock支持公平和不公平抢占,而synchronized是不公平锁;
6) Lock锁支持阻塞和不阻塞加锁,而synchronized锁支持阻塞加锁;
7) Lock可以设置超时机制,而synchronized不能设置超时;
8) Lock可以中断,而synchronized不可中断;
9) Lock底层原理是AQS,而synchronized是通过monitor对象监听器;

  • 5
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
synchronized代码块是一种在Java中用于实现线程同步的机制。使用synchronized关键字修饰的代码块,可以保证同一时刻只有一个线程可以进入该代码块执行。 代码块使用的是对象的方式,也就是说,的范围是一个特定的对象。通常情况下,我们使用this关键字作为对象,也就是定当前对象。当一个线程进入同步代码块时,它会持有该对象的,其他线程需要等待该线程释放后才能进入代码块执行。 另外,synchronized还可以非this对象。这种方式可以提高多线程并发执行的效率。例如,如果一个类中有多个synchronized方法,使用非this对象作为可以使得这些方法在执行时不互相阻塞,从而提高效率。 总结来说,synchronized代码块是一种用于实现线程同步的机制,可以确保同一时刻只有一个线程进入代码块执行。可以使用this关键字定当前对象,也可以使用非this对象作为来提高并发执行效率。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [synchronized方法, 代码块)](https://blog.csdn.net/luciferlongxu/article/details/124809818)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [synchronized机制 之 代码块](https://blog.csdn.net/qq_16504067/article/details/117695564)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值