synchronized的作用

一句话说出synchronized的作用:  能够保证在同一时刻最多只有一个线程执行该段代码,以达到保证并发安全效果。

大概意思: 当多个线程同时访问synchronized修饰的方法或者代码块的时候,要拿到这部分代码的锁才能访问,某一个线程拿到这个锁正在访问的时候,这部分代码是锁住的,其它线程会处于阻塞的状态,只能这个线程释放掉锁之后,其它的多个线程才会去竞争这个锁,拿到之后才能访问。

 synchronized的地位

  • synchronized是java的关键字
  • 是最基本的互斥同步手段
  • 是并发编程元老级角色,并发编程必懂知识

看看使用并发手段会有什么后果?(两个线程同时执行a++)


public class ThreadDemo3 implements Runnable{static int a = 0;/*** 1.重写run方法,在里面循环10W次,执行a++* 2.用两个线程执行,预期的结果输出结果 a=200000** @param args* @throws InterruptedException*/public static void main(String[] args) throws InterruptedException {ThreadDemo3 threadDemo3 = new ThreadDemo3();Thread t1 = new Thread(threadDemo3);Thread t2 = new Thread(threadDemo3);t1.start();t2.start();// 调用join方法,会让主线程处于等待状态,等到t1,t2线程执行完成以后才去执行t1.join();t2.join();System.out.println(threadDemo3.a);}@Overridepublic void run() {for (int i = 0; i < 100000; i++) {a++;}}
}

下面是我多次运行的结果,发现每次运行的结果必会比预期运行的少

这个是为什么那???

以为a++,看上去是一个操作,实际上包含了三个操作:

  1. 读取a
  2. 将a + 1
  3. 将a的值写入到内存中

在上述代码中对a值的操作没有同步,可能会出现 t1线程先读取a,这个时候a=6,在+1之后就是a=7,但是没有执行到第三步把最新的值写在内存中就切换了t2线程来执行,t2读取到值仍然是6。

我们使用synchronized关键字之后看看效果

这个时候控制台输出的就是我们预期的值了 。

说说synchronized的两个用法

  1. 对象锁:  包含了普通方法锁(锁对象当前实例对象) 和 同步代码块锁(指定的锁对象)
  2. 类锁: 静态方法锁和class对象锁(同步代码块中 .class)

对象锁----> 同步代码块示例


public class ThreadDemo5 implements Runnable{static Object lock = new Object();@Overridepublic void run() {// 使用对象locksynchronized (lock){
//          // 使用this 当前对象
//        synchronized (this){System.out.println("代码块锁 start , 线程名称: " + Thread.currentThread().getName());try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("代码块锁 end , 线程名称: " + Thread.currentThread().getName());}}/*** 对象锁   同步代码块示例* @param args*/public static void main(String[] args) {ThreadDemo5 temp = new ThreadDemo5();Thread t1 = new Thread(temp);Thread t2 = new Thread(temp);t1.start();t2.start();while (t1.isAlive() || t2.isAlive()){}System.out.println("主线程 end !!!");}
}

结果

对象锁---> 普通方法示例


public class ThreadDemo5 implements Runnable{@Overridepublic void run() {test();}/*** 方法加上  synchronized*/public synchronized void test(){System.out.println("普通方法锁 start , 线程名称: " + Thread.currentThread().getName());try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("普通方法锁 end , 线程名称: " + Thread.currentThread().getName());}public static void main(String[] args) {ThreadDemo5 temp = new ThreadDemo5();Thread t1 = new Thread(temp);Thread t2 = new Thread(temp);t1.start();t2.start();while (t1.isAlive() || t2.isAlive()){}System.out.println("主线程 end !!!");}

执行结果

切记 对象锁使用的锁是同一个对象,如果使用不同对象是没有同步的效果

类锁--->静态方法示例


public class ThreadDemo5 implements Runnable{@Overridepublic void run() {test();}/*** 静态方法加上  synchronized*/public synchronized static void test(){System.out.println("静态方法锁 start , 线程名称: " + Thread.currentThread().getName());try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("静态方法锁 end , 线程名称: " + Thread.currentThread().getName());}public static void main(String[] args) {// 使用两个实例ThreadDemo5 temp1 = new ThreadDemo5();ThreadDemo5 temp2 = new ThreadDemo5();Thread t1 = new Thread(temp1);Thread t2 = new Thread(temp2);t1.start();t2.start();while (t1.isAlive() || t2.isAlive()){}System.out.println("主线程 end !!!");}
}

执行结果

类锁-->同步代码块示例


public class ThreadDemo5 implements Runnable{@Overridepublic void run() {// ThreadDemo5.class 类锁synchronized (ThreadDemo5.class){System.out.println("代码块类锁 start , 线程名称: " + Thread.currentThread().getName());try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("代码块类锁 end , 线程名称: " + Thread.currentThread().getName());}}public static void main(String[] args) {// 使用两个实例ThreadDemo5 temp1 = new ThreadDemo5();ThreadDemo5 temp2 = new ThreadDemo5();Thread t1 = new Thread(temp1);Thread t2 = new Thread(temp2);t1.start();t2.start();while (t1.isAlive() || t2.isAlive()){}System.out.println("主线程 end !!!");}
}

// 输出结果

类锁同一个对象或两个对象都会达到同步效果

面试常问多线程访问同步方法的几种情况

1.两个线程同时访问一个对象的普通同步方法?

答: 具有同步效果

2.两个线程访问的是两个对象的普通同步方法?

答: 不具有同步效果,因为普通同步方法是对象锁,两个对象使用的锁对象不是同一个呀

3.两个线程访问的是synchronized的静态方法

答:具有同步效果,同步静态方法是类锁,锁的是类对象

4.两个线程访问一个对象同步方法与非同步方法

答: 会并行的执行,synchronized只作用在修饰的方法上面,非同步的方法是不受到影响的

5.访问同一个对象的不同的普通同步方法

答: 具有同步效果,普通同步方法锁都是同一个对象,所有会有同步效果

6.同时访问一个对象静态synchronized和非静态synchronized方法

答: 不具有同步效果,一个是类锁一个是对象锁,同时访问是不受到影响的

java中synchronized介绍和用法相关推荐

  1. java中synchronized的用法详解

    记下来,很重要. Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问同一个对象object中的这个synchron ...

  2. Java中Date各种相关用法

    Java中Date各种相关用法 本文主要介绍Java中Date各种相关用法. AD:   Java中Date各种相关用法(一) 1.计算某一月份的最大天数 Java代码 Calendar time=C ...

  3. Java 动态代理介绍及用法

    Java 动态代理介绍及用法 一,静态代理模式的特点 在之前的文章中 java代理模式 已经介绍里java里的(静态)代理模式 下面是上文静态代理类的例子: public class ProxyBea ...

  4. JAVA中 BufferedImage、ImageIO用法

    转载:JAVA中BufferedImage.ImageIO用法 - 邓维-java - 博客园 (cnblogs.com) BufferedImage          BufferedImage是其 ...

  5. java中iterator用法_java_详解Java中Iterator迭代器的用法,迭代器(Iterator)迭代 - phpStudy...

    详解Java中Iterator迭代器的用法 迭代器(Iterator) 迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构.迭代器通常被称为&quo ...

  6. Java中synchronized和volatile有什么区别?

    Java中synchronized和volatile有什么区别? 相关内容 synchronized的问题 什么叫做不完整对象,这个怎么理解呢? 总结 ) 相关内容 1.Java语言为了解决并发编程中 ...

  7. java 死锁 内存消耗_详解Java中synchronized关键字的死锁和内存占用问题

    先看一段synchronized 的详解: synchronized 是 java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并 ...

  8. java中arraycopy的用法_[jdk源码阅读系列]Java中System.arraycopy()的用法

    本文转载,原文链接: 3分钟了解Java中System.arraycopy的用法 - 伊万夫斯基 - 博客园  https://www.cnblogs.com/benjieqiang/p/114288 ...

  9. Java中split函数的用法及使用示例

    Java中split函数的用法及使用示例 2010-05-04 10:21 日志原文:http://lhgc.blog.sohu.com/80444801.html java.lang.string. ...

最新文章

  1. linux透明加密系统,基于Linux的透明加密系统的设计与实现
  2. HuggingFace-transformers系列的介绍以及在下游任务中的使用
  3. C语言结构体指针与结构体变量作形参的区别
  4. php 插入数据 不成功,thinkphp5连接oracle用insert插入数据失败
  5. 【POJ - 2406】Power Strings (KMP,最小循环节)
  6. 阶乘取模算法java_np问题(大数阶乘取模)
  7. 使用Phantom omni力反馈设备控制机器人
  8. 用Google App Engine做个人代理服务器
  9. html中th 与thead tbody的 使用
  10. LeetCode-25:K 个一组翻转链表
  11. 【论文写作】JSP旅游网如何写概念设计
  12. 【系统架构】缓存Memcache 使用原子性操作add,实现并发锁
  13. Yedda Twitter C# Library
  14. C派生matlab读取excel脚本
  15. 模拟集成电路设计初学系列
  16. send()、sendto()和recv()、recvfrom()的使用
  17. 飞到半路被撞?不是小鸟不专心,而是太阳能太晃眼
  18. CocosCreator 项目可编译debug无法编译release问题特例
  19. Adobe XD常见问题和解决方案
  20. 2023电信天翼云高级解决方案架构师考试题库-最新版

热门文章

  1. android x86小白安装教程,小白的教程,在windows上安装完美的Android手机驱动
  2. java优先队列的入队函数_Java内置的优先队列PriorityQueue
  3. 炉石整活拔线方法_炉石传说:采访仰天莫笑——黄金总决赛再度捧杯后的变化与成长...
  4. 路由与交换大作业pkt_干货 | 交换机“练功大法”——略有小成(一)
  5. oracle新建用户名和密码,Oracle安装后忘记用户名或密码+创建新登陆用户
  6. java servlet 请求_Java中前台JSP请求Servlet实例(http+Servlet)
  7. greenplum 存储过程_如何使用Greenplum提升PB级数据处理能力
  8. html财务统计,财务统计.html · 珠烟/layuiadmin-templete - Gitee.com
  9. 编写程序判断等腰、等边或者普通三角形
  10. 【c语言】蓝桥杯基础练习 闰年判断