一 什么是对象锁

对象锁也叫方法锁,是针对一个对象实例的,它只在该对象的某个内存位置声明一个标识该对象是否拥有锁,所有它只会锁住当前的对象,而并不会对其他对象实例的锁产生任何影响,不同对象访问同一个被synchronized修饰的方法的时候不会阻塞,
例如:

  1. public class MyObject {
  2. private synchronized void method1(){
  3. try {
  4. System.out.println(Thread.currentThread().getName());
  5. Thread.sleep(4000);
  6. } catch (InterruptedException e) {
  7. e.printStackTrace();
  8. }
  9. }
  10. //synchronized修饰为同步方法,如果先调用method1,则4秒后才会调用method2
  11. //如果不用synchronized修饰,则可以直接异步调用,没有影响
  12. private void method2(){
  13. System.out.println(Thread.currentThread().getName());
  14. }
  15. }

  创建一个类,synchronized修饰普通方法,即为对象锁,那么这个时候,多个线程访问同一个对象实例的这个方法时,是会同步的,并且只有一个线程执行完,另一个线程才会执行:

  1. public static void main(String[] args) {
  2. //创建一个对象
  3. MyObject myObject=new MyObject();
  4. Thread t1=new Thread (new Runnable() {
  5. @Override
  6. public void run() {
  7. myObject.method1();
  8. }
  9. },"t1");
  10. Thread t2=new Thread (new Runnable() {
  11. @Override
  12. public void run() {
  13. myObject.method1();
  14. }
  15. },"t2");
  16. t1.start();
  17. t2.start();
  18. }

  即,打印t14秒之后,t2才会打印,因为两个线程调用的是同一个对象实例的方法,即同一把锁,所有会同步执行
而如果是不同对象实例的话,则没有影响,因为两个线程调用的是不同实例的锁方法,即不是同一把锁,没有关系,所以会正常输出,不会同步

  1. public static void main(String[] args) {
  2. //创建两个对象
  3. MyObject myObject=new MyObject();
  4. MyObject myObject01=new MyObject();
  5. Thread t1=new Thread (new Runnable() {
  6. @Override
  7. public void run() {
  8. myObject.method1();
  9. }
  10. },"t1");
  11. Thread t2=new Thread (new Runnable() {
  12. @Override
  13. public void run() {
  14. myObject01.method1();
  15. }
  16. },"t2");
  17. t1.start();
  18. t2.start();
  19. }

二 对象锁的几种形式以及应用案例

1 synchronized修饰普通方法属于对象锁,
2,java对象锁与类锁 - 图1
synchronized修饰的代码块传入this也属于对象锁
应用:减小锁粒度,第二种形式就比较好,比如A线程调用一个同步方法需要很长时间,那么B就要等待很长时间,这个时候可以将必须同步的代码使用synchronized代码块,
java对象锁与类锁 - 图2

不不需要同步的先执行,节约资源

三 类锁

类锁是锁住整个类,当有多个线程来声明这个类的对象时候将会被阻塞,直到拥有这个类锁的对象呗销毁或者主动释放了类锁,这个时候在被阻塞的线程被挑选出一个占有该类锁,声明该类的对象。其他线程继续被阻塞住
(上面百度的),即一句话,不管多少个对象,多少个对象,共用一把多,且只有一把,不管怎么调用,都会同步
上面方法加static变类锁:

  1. private static synchronized void method1(){
  2. try {
  3. System.out.println(Thread.currentThread().getName());
  4. Thread.sleep(4000);
  5. } catch (InterruptedException e) {
  6. e.printStackTrace();
  7. }
  8. }

  这个时候无论线程调用的是多少个对象实例的方法,都会同步

四 类锁形式

1 synchronized修饰静态方法属于类锁
2java对象锁与类锁 - 图3