synchronized和lock锁的区别?

1 Synchronized

1.1 可以修饰的位置

synchronized可以修饰静态方法、非静态方法、代码块、一个对象、一个类。

  1. synchronized关键字也可以修饰静态方法,此时如果调用该静态方法,将会锁住整个类,锁是当前类的class对象
  2. 锁的都是synchronized 关键字后面的大括号包起来的部分。这一部分是不能被多线程同时访问的。
  3. synchronized修饰普通同步方法锁是当前实例对象
  4. 同步方法块,锁是括号里面的对象

1.2 Synchronized是如何实现同步(实现原理)

同步代码块

同步代码块是使用monitorenter和monitorexit指令实现的,

  1. monitorenter指令是在编译后插入到同步代码块的开始位置
  2. monitorexit指令插入到同步代码块的结束位置,
  3. JVM需要保证每一个monitorenter都有一个monitorexit与之相对应

同步方法

同步方法(在这看不出来需要看JVM底层实现)依靠的是方法修饰符上的ACC_SYNCHRONIZED实现

  1. JVM就是根据该标示符(ACC_SYNCHRONIZED)来实现方法的同步的:
  2. 方法调用时,调用指令将会检查方法的 ACC_SYNCHRONIZED 访问标志是否被设置
  3. 如果设置了,执行线程将先获取monitor获取成功之后才能执行方法体,方法执行完后再释放monitor

2 lock和synchronized的区别

2.0 所有区别

1、lock是一个接口synchronized是java的一个关键字,是在jvm的层面上

2、synchronized在发生异常时候会自动释放占有的锁,因此不会出现死锁;而lock发生异常时候,不会主动释放占有的锁,必须手动unlock来释放锁可能引起死锁的发生

使用lock时最好是将同步代码块用try catch包起来,finally中写入unlock,及时释放锁,避免死锁的发生。

3、lock等待锁过程中可以用interrupt来中断等待,而synchronized只能等待锁的释放,不能响应中断

4、Lock可以通过trylock来知道有没有获取锁,而synchronized不能;

5、Lock可以提高多个线程进行读操作的效率

6、如果竞争资源不激烈,两者的性能是差不多的;而当竞争资源非常激烈时(即有大量线程同时竞争),此时Lock的性能要远远优于synchronized

7、synchronized使用Object对象本身的wait 、notify、notifyAll调度机制,而Lock可以使用Condition进行线程之间的调度

  1. Condition是个接口,基本的方法就是await()和signal()方法
  2. Condition依赖于Lock接口,生成一个Condition的基本代码是lock.newCondition()
  3. Condition的await()和signal()方法,都必须在lock保护之内,就是说必须在lock.lock()和lock.unlock之间才可以使用
    • Conditon中的await()对应Object的wait()
    • Condition中的signal()对应Object的notify()
    • Condition中的signalAll()对应Object的notifyAll()
//Condition是个接口,基本的方法就是await()和signal()方法;
Lock lock=new ReentrantLock();
Condition condition=lock.newCondition();
...
condition.await();
...
condition.signal();
condition.signalAll();


2.1 用法区别

synchronized:在需要同步的对象中加入此控制,synchronized可以加在方法上,也可以加在特定代码块中,括号中表示需要锁的对象。

lock:一般使用ReentrantLock类做为锁。在加锁和解锁处需要通过lock()和unlock()显示指出。所以一般会在finally块中写unlock()以防死锁

2.2 性能区别

synchronized是托管给JVM执行的,

lock是java写的控制锁的代码

Java1.5中,synchronize是性能低效的,没有lock性能好

Java1.6上synchronize的性能并不比Lock差。

synchronized原始采用的是CPU悲观锁机制,即线程获得的是独占锁。

其他线程只能依靠阻塞来等待线程释放锁,CPU转换线程阻塞时会引起线程上下文切换,当有很多线程竞争锁的时候,会引起CPU频繁的上下文切换导致效率很低

Lock用的是乐观锁方式。

  1. 所谓乐观锁就是,每次不加锁而是假设没有冲突而去完成某项操作,如果因为冲突失败就重试,直到成功为止
  2. 乐观锁实现的机制就是CAS操作

2.3 lock锁原理的简单总结

lock的存储结构双链表(用于存储等待中的线程)+int类型的状态值(用于锁的状态变更)

lock获得锁的过程:本质通过CAS来更改状态值(0,1)若线程没取到,则进入等待队列(等待链表)中

lock释放锁修改状态值,调整等待的链表

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
synchronizedlockJava中用于实现线程同步的两种不同机制。 1. synchronized关键字是Java语言提供的内置机制,用于保证代码块或方法在同一时刻只能由一个线程执行。synchronized关键字可以用于修饰代码块或方法,当一个线程进入synchronized代码块或方法时,会自动获取,并在执行完毕后释放synchronized关键字的是隐式的,由Java虚拟机自动管理。 2. Lock接口是Java.util.concurrent包提供的显式机制,也是一种更灵活、可控制性更强的机制。Lock接口的实现类可以通过调用lock()方法获取,并通过调用unlock()方法释放。与synchronized不同,Lock接口可以实现更细粒度的定,并提供了更多高级功能,如可重入、读写等。 下面是synchronizedLock之间的一些区别: - 可重入性:synchronized是可重入,即一个线程可以多次获取同一个;而Lock接口可以通过实现ReentrantLock类来实现可重入。 - 的获取方式:synchronized关键字是隐式的,在进入synchronized代码块或方法时自动获取,并在退出时释放;而Lock接口需要显式地调用lock()方法获取,并在finally块中调用unlock()方法释放。 - 等待可中断:通过Lock接口提供的lockInterruptibly()方法,可以在等待获取的过程中响应中断请求,而synchronized关键字在等待获取时无法响应中断。 - 公平性:Lock接口可以实现公平,即按照线程请求的顺序来获取,而synchronized关键字是非公平。 - 性能:在低竞争情况下,synchronized关键字的性能表现更好;而在高竞争情况下,Lock接口的性能更好。 总的来说,synchronized是一种简单易用的线程同步机制,适用于大部分场景;而Lock接口提供了更多灵活、可控制的定方式,适用于一些特殊需求的场景。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

?abc!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值