2023-01-14  阅读(3)
原文作者: HelloWorld_EE 原文地址:https://blog.csdn.net/u010412719/category_6159934_2.html

Java中synchronized关键字的使用

对于关键字synchronized,研究起来,发现还是有许多让自己模糊的地方,网上也有很多篇博客对synchronized关键字的使用讲解的相当好,自己也受益匪浅。自己之所以还写一篇博客来介绍synchronized的目的只有一个:加深自己对synchronized的理解。

写博客有时候确实是一个好的东西,往往研究某个知识点的时候,自己觉得弄懂了,但是过几天查不多就忘了模糊了,而写博客可以增加对知识点的理解和加深知识的记忆。

synchronized关键字是解决多线程并发同步的方法之一。

synchronized可以修饰如下几个方面。

1、修饰一个代码块,作用的对象是调用这个代码块的对象

2、修饰一个方法,作用的对象是调用这个方法的对象

3、修饰一个静态方法,作用的对象是静态方法所属的类的所有对象

4、修饰一个类,作用的对象是该类的所用对象。

下面一一进行介绍

1、synchronized修饰一个代码块

1.1 一个线程访问一个对象obj中的synchronize(this)同步代码块时,其它线程试图访问该对象obj的synchronize(this)同步块时将会被阻塞。

看下面这个例子

Demo1

        class MyThread implements Runnable{
            private static final int NUM = 3;
            @Override
            public void run() {
                synchronized(this){
                    for(int i =0;i<NUM;i++){
                        System.out.println(Thread.currentThread().getName()+"  running .....");
                        try {
                            Thread.sleep(100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
    
                }
            }
    
        }

测试代码如下:

        public class SyncCodeBlock {
    
            public static void main(String[] args) {
                MyThread mt = new MyThread();
                new Thread(mt,"Thread1").start();
                new Thread(mt,"Thread2").start();
            }
    
        }

运行结果:

    Thread2  running .....
    Thread2  running .....
    Thread2  running .....
    Thread1  running .....
    Thread1  running .....
    Thread1  running .....

这里,名字为Thread1、Thread2的两个线程都想访问对象mt的同步块synchronized(this),由于只有一个锁,谁拿到的这个锁就该谁访问,也就是说,在任一时刻只能有一个线程访问这一同步代码块。例如,当Thread1线程拿到锁正在执行run里面的代码时,名字为Thread2的线程也想访问同一对象mt的同步块synchroized(this)的同步块将会被阻塞,只有等Thread1访问结束之后释放该对象锁Thread2拿到该对象锁才能访问。

如果我们将测试代码该为如下:

        public static void main(String[] args){
            MyThread mt = new MyThread();
            MyThread mt2 = new MyThread();
            new Thread(mt,"Thread1").start();
            new Thread(mt2,"Thread2").start();
        }

结果如下:

    Thread2  running .....
    Thread1  running .....
    Thread2  running .....
    Thread1  running .....
    Thread2  running .....
    Thread1  running .....

这里,和上面的第一种测试代码不同,这里有两个不同的MyThread对象,分别对应着两把锁,因此线程Thread1访问同步块是去拿对象mt的锁,而线程Thread2访问同步块是去拿对象mt2的锁,而这两个锁是没有任何关系的,即线程Thread1执行的是对象mt的中的synchronized代码块,而线程Thread2执行的是对象mt2中的synchronized代码块,两者互不相关,因此这两个线程就可以同时进行。

小结:当一个线程访问一个对象obj中的synchronized(this)同步块时,其它线程访问这个对象obj的synchronize(this)就会阻塞

1.2 当一个线程访问一个对象obj中的synchronized(this)同步代码块时,其它的线程可以同时访问该对象obj中的非synchronize(this)代码块

看一个例子

Demo2:多个线程访问synchronized和非synchronized代码块

        class MyThread implements Runnable{
            private static final int NUM = 3;
            @Override
            public void run() {
                String name = Thread.currentThread().getName();
                if(name.equals("Thread1")){
                    synMethod();
                }
                else if(name.equals("Thread2")){
                    notSynMethod();
                }
            }
            public void synMethod(){
                synchronized(this){
                    for(int i=0;i<NUM;i++){
                        System.out.println(Thread.currentThread().getName()+"running ....");
                        try {
                            Thread.sleep(100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                }
            }
            public void notSynMethod(){
                for(int i=0;i<NUM;i++){
                    System.out.println(Thread.currentThread().getName()+"running ....");
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
    
        }

测试代码:

    
        public static void main(String[] args) {
            MyThread mt = new MyThread();
            new Thread(mt,"Thread1").start();
            new Thread(mt,"Thread2").start();
        }

运行结果:

    Thread1 running ....
    Thread2 running ....
    Thread1 running ....
    Thread2 running ....
    Thread1 running ....
    Thread2 running ....

分析:

上面的代码中,有一个使用synchronize(this)同步的方法synMethod,有一个没有使用关键字synchronize(this)同步的方法notSynMethod.

由于线程Thread1和Thread2没有同时访问对象mt的synchronized(this)同步代码块,而是只有Thread1访问,Thread2访问的是对象mt非synchronized(this)代码块。因此这两个线程是可以同时进行的。

注意:这里的“访问对象mt的非synchronized(this)代码块”并不是单单指对象mt中没有用关键字synchronized修饰的代码块,而是还包括对象mt中其它使用类似synchronized(otherRef)修饰的代码块,其中otherRef为不等于this的其它引用对象。

看下面的例子,将上例Demo2中的notSynMethod方法用synchronized(otherRefe)来同步

        private int [] value = new int[0];
        public void notSynMethod(){
            synchronized(value){
                for(int i=0;i<NUM;i++){
                    System.out.println(Thread.currentThread().getName()+" running ....");
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
    
        }

改成这样,线程Thread1、Thead2依然可以同时工作,这是因为他们所需要的锁不一样。

小结:当一个线程访问对象obj的synchronized(this)代码块时,其它线程是可以同时访问该对象obj的非synchronized(this)的代码块

1.3、当多线程并发时,一个线程访问对象obj的synchronized(this)代码块时,其它线程对对象obj中所有其他synchronized(this)同步代码块的访问将被阻塞

看一个例子:MyThread5类中的两个方法都使用了synchronized(this)来进行修饰.

        class MyThread5 implements Runnable{
            private static final int NUM = 3;
            @Override
            public void run() {
                String name = Thread.currentThread().getName();
                if(name.equals("Thread1")){
                    synMethod();
                }
                else if(name.equals("Thread2")){
                    synMethod2();
                }
            }
            public void synMethod(){
                synchronized(this){
                    for(int i=0;i<NUM;i++){
                        System.out.println(Thread.currentThread().getName()+" running ....");
                        try {
                            Thread.sleep(100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                }
            }
            public void synMethod2(){
                synchronized(this){
                    for(int i=0;i<NUM;i++){
                        System.out.println(Thread.currentThread().getName()+" running ...." + i);
                        try {
                            Thread.sleep(10);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                }
    
            }
    
        }

测试方法为:

        public static void main(String[] args) {
            MyThread5 mt = new MyThread5();
            new Thread(mt,"Thread1").start();
            new Thread(mt,"Thread2").start();
        }

运行结果:

    Thread2 running ....0
    Thread2 running ....1
    Thread2 running ....2
    Thread1 running ....
    Thread1 running ....
    Thread1 running ....

分析:

当线程Thread1访问对象mt的方法synMethod中的synchronized(this)同步代码块时,线程Thread2访问对象mt中另一个方法synMethod2中synchronized(this)同步代码块的访问将被阻塞。这是因为,线程Thread1访问mt的一个synchronized(this)同步代码块时,它就获得了这个mt的对象锁。结果,线程Thread2对该mt对象所有使用synchronized(this)的同步代码部分的访问都被暂时阻塞。

2、synchronized修饰某个方法

synchronized修饰某个方法,作用的对象为调用该方法的对象。

其实,

        public synchronized void method(){
            //to do something ....
        }

效果等同于:

        public void method(){
            synchronized(this){
                //to do something ....
            }
        }

因此,这里不再进行介绍。

不过,有两点需要注意的是

1、一般我们不建议同步整个方法,能同步方法中的某个代码块就同步代码块。同步安全往往是以性能为代价的。

2、synchronized关键字同步的方法不能继承。虽然可以使用synchronized来修饰某个方法,但是synchronized并不属于方法定义的一部分,因此,synchronized关键字不能被继承。如果在父类中的某个方法使用了synchronized关键字,而在子类中覆盖了这个方法,在子类中的这个方法默认情况下并不是同步的,而必须显式地在子类的这个方法中加上synchronized关键字才可以。

3.synchronized关键字修饰静态代码块

synchronized关键字修饰静态代码块,作用的是该类的所用对象。

Demo3:synchronized关键字修饰静态代码块

        class MyThread2 implements Runnable{
    
            private static final int NUM = 3;
            @Override
            public void run() {
                print();
            }
    
            public synchronized static void print() {
                for(int i=0;i<NUM;i++){
                    System.out.println(Thread.currentThread().getName()+" running... "+i);
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
    
        }

测试代码:

        public static void main(String[] args) {
            MyThread2 mt = new MyThread2();
            MyThread2 mt2 = new MyThread2();
            new Thread(mt,"Thread1").start();
            new Thread(mt2,"Thread2").start();
    
        }

分析:

我们都知道静态方法是属于类的,不是属于对象的。

MyThread2类中的静态方法print使用了synchronized修饰,尽管在测试代码中使用了两个不同的对象mt/mt2,但是这里的锁不再是锁对象mt/mt2.而是锁”类MyThread2”这个对象,mt/mt2都是属性类MyThread的,因此mt/mt2就相当于同一把锁。,因此,这两个线程在访问此静态方法print时就需要取得锁之后才能访问,访问完之后释放锁。

4、synchronized修饰一个类

synchronized修饰一个类,作用为该类的所用对象

        class MyThread3 implements Runnable{
    
            private static final int NUM = 3;
            @Override
            public void run() {
                synchronized(MyThread3.class){
                    for(int i=0;i<NUM;i++){
                        System.out.println(Thread.currentThread().getName()+" running... "+i);
                        try {
                            Thread.sleep(100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                }
            }
    
        }

测试代码如下:

        public static void main(String[] args) {
            MyThread3 mt = new MyThread3();
            MyThread3 mt2 = new MyThread3();
            new Thread(mt,"Thread1").start();
            new Thread(mt2,"Thread2").start();
        }

运行结果如下:

    Thread2 running... 0
    Thread2 running... 1
    Thread2 running... 2
    Thread1 running... 0
    Thread1 running... 1
    Thread1 running... 2

由于mt/mt2都是属于类MyThread3的,而synchronized修饰的是MyThread3这整个类,因此所有的MyThread访问此代码块都是互斥的,任一时刻都只能有一个线程能够访问。

不知道大家有没有这样的疑问,反正我是有的,既然synchronized修饰的静态方法和修饰的整个类都是作用与该类的全部对象,那么这两者是不是互斥的呢??

上面的问题具体一点来描述:有一个MyThread类,MyThread类中的一个方法中的代码块Block1用synchronize(MyThread.class)同步,MyThread类中一个静态方法method用synchronized同步,有两个线程Thread1、Thread2,Thread1访问Block1,Thread2访问method是否可以同时进行???

答案是:可以的,这是因为虽然都作用与该类的所用对象,但是锁却不是同一个锁,因此不是互斥的,可以同时访问

看如下的例子

        class MyThread4 implements Runnable{
    
            private static final int NUM = 3;
            @Override
            public void run() {
                String name = Thread.currentThread().getName();
                if(name.equals("Thread1")){
                    synClassMethod();
                }
                else if(name.equals("Thread2")){
                    synStaticCodeBlock();
                }
            }
    
            public void synClassMethod(){
                synchronized(MyThread4.class){
                    for(int i=0;i<NUM;i++){
                        System.out.println(Thread.currentThread().getName()+" running... "+i);
                        try {
                            Thread.sleep(100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                }
            }
            public static void synStaticCodeBlock(){    
                for(int i=0;i<NUM;i++){
                    System.out.println(Thread.currentThread().getName()+" running... "+i);
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
    
            }
    
        }

测试代码如下:

        public static void main(String[] args) {
            MyThread4 mt = new MyThread4();
            MyThread4 mt2 = new MyThread4();
            new Thread(mt,"Thread1").start();
            new Thread(mt2,"Thread2").start();
        }

运行结果为:

    Thread2 running... 0
    Thread1 running... 0
    Thread2 running... 1
    Thread1 running... 1
    Thread2 running... 2
    Thread1 running... 2

总结

1、当两个线程Thread1、Thread2访问一个对象obj的synchronized(this)代码块时,每个时刻都只能有一个访问此代码块,当Thread1访问时,Thread2只有在Thread1线程执行完这段代码块并释放锁后取得锁才能访问。

2、当两个线程并发时,一个线程访问对象obj的synchronized(this)代码块时,其它的线程可以访问对象obj的非synchronized(this)代码块。

3、当多线程并发时,一个线程访问对象obj的synchronized(this)代码块时,其它线程对对象obj中所有其他synchronized(this)同步代码块的访问将被阻塞。这是因为,当一个线程访问obj的一个synchronized(this)同步代码块时,它就获得了这个obj的对象锁。结果,其它线程对该obj对象所有同步代码部分的访问都被暂时阻塞。

4、第3个结论同样适用其它同步代码块。例如,如果一个线程访问对象lock的synchronized(lock)的同步代码块,其中private int[] lock = new int[0],则其它线程在在其它方法中出现的
synchronized(lock)的同步代码块的访问将被阻塞。

5、上面所有的结论都适用于其它对象锁。

参考资料

1、http://blog.csdn.net/luoweifu/article/details/46613015

2、http://www.cnblogs.com/GnagWang/archive/2011/02/27/1966606.html


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文