ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

Java进阶:synchronized使用详解

2021-02-06 10:29:34  阅读:181  来源: 互联网

标签:10 SynchronizedTest Java 进阶 synchronized Thread 锁住 线程 new


1. synchronized的作用

锁住代码块,保证同一段代码在某一时刻只能有一个线程执行,该线程执行完毕后,其他线程再执行。目的是防止多线程并发操作导致的异常。

例如:当前有个变量count=0,线程A和线程B同时执行如下代码:

if (count == 0){
	count = count + 10;
}

存在这种情况,A执行了if (count == 0) 符合条件,此时B也执行了if语句同样符合条件,然后两个线程都执行 count = count + 10 。最终count=20

2. synchronized几种使用方式

  • 锁住普通对象:锁的实例是对象,不同线程lock同一个对象需要互相等待,lock不同对象则不影响

  • 锁住静态对象:由于是静态对象,所以对象永远是同一个,多个线程操作一定会互相等待

  • 锁住普通方法:效果等同于锁住普通对象,举个例子说明

    public synchronized void lockMethod() {
    	//此处等待3秒,如果被锁住了,则其他线程的等待时间是叠加的
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName() + ":" + new Date());
    }
    
    public static void main(String[] args) {
    	//用同一个对象操作
        SynchronizedTest obj = new SynchronizedTest();
        new Thread(() -> {
            obj.lockMethod();
        }, "Thread-1").start();

        new Thread(() -> {
            obj.lockMethod();
        }, "Thread-2").start();
    }

输出结果如下,时间是叠加的,说明synchronized起效果:

Thread-1:Sat Feb 06 10:10:13 CST 2021
Thread-2:Sat Feb 06 10:10:16 CST 2021

main 方法改一下,调用不同对象的方法:

    public static void main(String[] args) {
        new Thread(() -> {
            new SynchronizedTest().lockMethod();
        }, "Thread-1").start();

        new Thread(() -> {
            new SynchronizedTest().lockMethod();
        }, "Thread-2").start();
    }

输出结果如下,两个时间基本一致,说明synchronized并没有互相阻塞,因为lock的不同对象:

Thread-2:Sat Feb 06 10:11:38 CST 2021
Thread-1:Sat Feb 06 10:11:38 CST 2021
  • 锁住静态方法:效果等同于锁住静态变量,也等同于锁住当前类的class
    public synchronized static void lockStaticMethod() {
    	//此处同样等待3秒
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName() + ":" + new Date());
    }
    
    public static void main(String[] args) {
        new Thread(() -> {
            SynchronizedTest.lockStaticMethod();
        }, "Thread-1").start();

        new Thread(() -> {
            SynchronizedTest.lockStaticMethod();
        }, "Thread-2").start();
    }

输出结果如下:

Thread-1:Sat Feb 06 10:14:01 CST 2021
Thread-2:Sat Feb 06 10:14:04 CST 2021

还有另一种写法,锁住class,效果也是一样的:

    public void lockClass() {
        synchronized (SynchronizedTest.class){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + ":" + new Date());
        }
    }
    
    public static void main(String[] args) {
        new Thread(() -> {
            new SynchronizedTest().lockClass();
        }, "Thread-1").start();

        new Thread(() -> {
            new SynchronizedTest().lockClass();
        }, "Thread-2").start();
    }

标签:10,SynchronizedTest,Java,进阶,synchronized,Thread,锁住,线程,new
来源: https://blog.csdn.net/qq_28834355/article/details/113705918

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有