深入理解synchronized(synchronized锁住的是代码还是对象)

2023-12-01 12:08

本文主要是介绍深入理解synchronized(synchronized锁住的是代码还是对象),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

面试安卓难免会问到许多的java问题,毕竟安卓目前就是用java编写的,虽然一些小众语言很强势,比如kotlin,AndroidStudio的一个插件,可以用来开发安卓程序,kotlin有许多特性是java所没有的,比如空指针的问题,好像有点跑偏了,根据我的面试经历,今天来说下Synchornized的理解及面试题。

面试官问你,考虑到并发场景,如何实现线程同步,你一想,这不就是Java中Sychronized关键字的具体使用嘛,你回答,使用同步 sychronized关键字,然后面试官又问了,具体怎么实现,你可能会回答,一般来说是同步方法,和同步代码块用的比较多(反正我是这样回答的,这样回答也是没问题的),下面这篇博客我看讲的还算是挺细的,能够结合具体实例来说明,博客地址为:http://www.cnblogs.com/QQParadise/articles/5059824.html,谢谢这个博主的讲解,大家要多理解,写代码来验证下!

在Java中,synchronized关键字是用来控制线程同步的,就是在多线程的环境下,控制synchronized代码段不被多个线程同时执行。synchronized既可以加在一段代码上,也可以加在方法上。

关键是,不要认为给方法或者代码段加上synchronized就万事大吉,看下面一段代码:

  1.   
  2.     public synchronized void test() {  
  3.         System.out.println("test开始..");  
  4.         try {  
  5.             Thread.sleep(1000);  
  6.         } catch (InterruptedException e) {  
  7.             e.printStackTrace();  
  8.         }  
  9.         System.out.println("test结束..");  
  10.     }  
  11. }  
  12.   
  13. class MyThread extends Thread {  
  14.   
  15.     public void run() {  
  16.         Sync sync = new Sync();  
  17.         sync.test();  
  18.     }  
  19. }  
  20.   
  21. public class Main {  
  22.   
  23.     public static void main(String[] args) {  
  24.         for (int i = 0; i < 3; i++) {  
  25.             Thread thread = new MyThread();  
  26.             thread.start();  
  27.         }  
  28.     }  
  29. }  
class Sync {public synchronized void test() {System.out.println("test开始..");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("test结束..");}
}class MyThread extends Thread {public void run() {Sync sync = new Sync();sync.test();}
}public class Main {public static void main(String[] args) {for (int i = 0; i < 3; i++) {Thread thread = new MyThread();thread.start();}}
}

 

运行结果: test开始.. test开始.. test开始.. test结束.. test结束.. test结束..

可以看出来,上面的程序起了三个线程,同时运行Sync类中的test()方法,虽然test()方法加上了synchronized,但是还是同时运行起来,貌似synchronized没起作用。 

将test()方法上的synchronized去掉,在方法内部加上synchronized(this):

  1. public void test() {  
  2.     synchronized(this){  
  3.         System.out.println("test开始..");  
  4.         try {  
  5.             Thread.sleep(1000);  
  6.         } catch (InterruptedException e) {  
  7.             e.printStackTrace();  
  8.         }  
  9.         System.out.println("test结束..");  
  10.     }  
  11. }  
public void test() {synchronized(this){System.out.println("test开始..");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("test结束..");}
}

 

运行结果: test开始.. test开始.. test开始.. test结束.. test结束.. test结束..

一切还是这么平静,没有看到synchronized起到作用。 

实际上,synchronized(this)以及非static的synchronized方法(至于static synchronized方法请往下看),只能防止多个线程同时执行同一个对象的同步代码段。

回到本文的题目上:synchronized锁住的是代码还是对象。答案是:synchronized锁住的是括号里的对象,而不是代码。对于非static的synchronized方法,锁的就是对象本身也就是this。

当synchronized锁住一个对象后,别的线程如果也想拿到这个对象的锁,就必须等待这个线程执行完成释放锁,才能再次给对象加锁,这样才达到线程同步的目的。即使两个不同的代码段,都要锁同一个对象,那么这两个代码段也不能在多线程环境下同时运行。

所以我们在用synchronized关键字的时候,能缩小代码段的范围就尽量缩小,能在代码段上加同步就不要再整个方法上加同步。这叫减小锁的粒度,使代码更大程度的并发。原因是基于以上的思想,锁的代码段太长了,别的线程是不是要等很久,等的花儿都谢了。当然这段是题外话,与本文核心思想并无太大关联。

再看上面的代码,每个线程中都new了一个Sync类的对象,也就是产生了三个Sync对象,由于不是同一个对象,所以可以多线程同时运行synchronized方法或代码段。

为了验证上述的观点,修改一下代码,让三个线程使用同一个Sync的对象。

  1. class MyThread extends Thread {  
  2.   
  3.     private Sync sync;  
  4.   
  5.     public MyThread(Sync sync) {  
  6.         this.sync = sync;  
  7.     }  
  8.   
  9.     public void run() {  
  10.         sync.test();  
  11.     }  
  12. }  
  13.   
  14. public class Main {  
  15.   
  16.     public static void main(String[] args) {  
  17.         Sync sync = new Sync();  
  18.         for (int i = 0; i < 3; i++) {  
  19.             Thread thread = new MyThread(sync);  
  20.             thread.start();  
  21.         }  
  22.     }  
  23. }  
class MyThread extends Thread {private Sync sync;public MyThread(Sync sync) {this.sync = sync;}public void run() {sync.test();}
}public class Main {public static void main(String[] args) {Sync sync = new Sync();for (int i = 0; i < 3; i++) {Thread thread = new MyThread(sync);thread.start();}}
}

 

运行结果: test开始.. test结束.. test开始.. test结束.. test开始.. test结束..

可以看到,此时的synchronized就起了作用。 

那么,如果真的想锁住这段代码,要怎么做?也就是,如果还是最开始的那段代码,每个线程new一个Sync对象,怎么才能让test方法不会被多线程执行。 

解决也很简单,只要锁住同一个对象不就行了。例如,synchronized后的括号中锁同一个固定对象,这样就行了。这样是没问题,但是,比较多的做法是让synchronized锁这个类对应的Class对象。

  1. class Sync {  
  2.   
  3.     public void test() {  
  4.         synchronized (Sync.class) {  
  5.             System.out.println("test开始..");  
  6.             try {  
  7.                 Thread.sleep(1000);  
  8.             } catch (InterruptedException e) {  
  9.                 e.printStackTrace();  
  10.             }  
  11.             System.out.println("test结束..");  
  12.         }  
  13.     }  
  14. }  
  15.   
  16. class MyThread extends Thread {  
  17.   
  18.     public void run() {  
  19.         Sync sync = new Sync();  
  20.         sync.test();  
  21.     }  
  22. }  
  23.   
  24. public class Main {  
  25.   
  26.     public static void main(String[] args) {  
  27.         for (int i = 0; i < 3; i++) {  
  28.             Thread thread = new MyThread();  
  29.             thread.start();  
  30.         }  
  31.     }  
  32. }  
class Sync {public void test() {synchronized (Sync.class) {System.out.println("test开始..");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("test结束..");}}
}class MyThread extends Thread {public void run() {Sync sync = new Sync();sync.test();}
}public class Main {public static void main(String[] args) {for (int i = 0; i < 3; i++) {Thread thread = new MyThread();thread.start();}}
}

 

运行结果: test开始.. test结束.. test开始.. test结束.. test开始.. test结束..

上面代码用synchronized(Sync.class)实现了全局锁的效果。

static synchronized方法,static方法可以直接类名加方法名调用,方法中无法使用this,所以它锁的不是this,而是类的Class对象,所以,static synchronized方法也相当于全局锁,相当于锁住了代码段。


这篇关于深入理解synchronized(synchronized锁住的是代码还是对象)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/441089

相关文章

HTML5实现的移动端购物车自动结算功能示例代码

《HTML5实现的移动端购物车自动结算功能示例代码》本文介绍HTML5实现移动端购物车自动结算,通过WebStorage、事件监听、DOM操作等技术,确保实时更新与数据同步,优化性能及无障碍性,提升用... 目录1. 移动端购物车自动结算概述2. 数据存储与状态保存机制2.1 浏览器端的数据存储方式2.1.

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,

Python如何去除图片干扰代码示例

《Python如何去除图片干扰代码示例》图片降噪是一个广泛应用于图像处理的技术,可以提高图像质量和相关应用的效果,:本文主要介绍Python如何去除图片干扰的相关资料,文中通过代码介绍的非常详细,... 目录一、噪声去除1. 高斯噪声(像素值正态分布扰动)2. 椒盐噪声(随机黑白像素点)3. 复杂噪声(如伪

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使

一文深入详解Python的secrets模块

《一文深入详解Python的secrets模块》在构建涉及用户身份认证、权限管理、加密通信等系统时,开发者最不能忽视的一个问题就是“安全性”,Python在3.6版本中引入了专门面向安全用途的secr... 目录引言一、背景与动机:为什么需要 secrets 模块?二、secrets 模块的核心功能1. 基

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

MySQL JSON 查询中的对象与数组技巧及查询示例

《MySQLJSON查询中的对象与数组技巧及查询示例》MySQL中JSON对象和JSON数组查询的详细介绍及带有WHERE条件的查询示例,本文给大家介绍的非常详细,mysqljson查询示例相关知... 目录jsON 对象查询1. JSON_CONTAINS2. JSON_EXTRACT3. JSON_TA