今天一起来认识认识 synchronized 这个一面试就会被提到的关键字。这一篇不会讲太多理论,主要先熟悉熟悉一下最简单的用法。只讨论一个问题:方法没用 synchronized 和用了 synchronized 有什么区别?

首先我们的理论基础是 synchronized 关键字是用在多线程并发的场景,所以接下来的测试代码都会模拟多线程并发的情况。我们直接通过代码来给大家讲讲加没加 synchronized 的区别。

1. 方法没加 synchronized 关键字

我们先写一个普通到不能再普通的代码,就是一个实例方法,在多线程并发的情况下执行同一个方法。如下所示。

public class NoSynchronizedTest {

    public static void main(String[] args) {
NoSynchronizedTest noSynchronizedTest = new NoSynchronizedTest();
for (int i = 0; i < 5; i ++) {
Thread thread = new Thread(() -> {
noSynchronizedTest.testNoSynchronizedMethod();
});
thread.start();
}
} public void testNoSynchronizedMethod() {
System.out.println("testNoSynchronizedMethod-start-" + Thread.currentThread().getName());
System.out.println("testNoSynchronizedMethod-end-" + Thread.currentThread().getName());
} }

运行结果:

我们可以发现如下结论:

  1. 在单线程中,执行顺序是有保证的,也就是每个 start 都优先于 end。
  2. 在多线程中,线程之间的 start 和 end 是没有先后顺序的,先进入方法的线程不一定先执行完方法。我们看运行结果的红色框,5 个线程只有 Thread-2 是完整的执行完方法,没有其他线程乱入的。

2. 方法加 synchronized 关键字

我们对上面的代码稍加修改,在方法上加 synchronized 修饰,并且在两种情况下执行代码。

5 个线程只有一个 synchronizedTest 对象。

public class SynchronizedTest {

    public static void main(String[] args) {
SynchronizedTest synchronizedTest = new SynchronizedTest();
for (int i = 0; i < 5; i++) {
Thread thread = new Thread(() -> {
synchronizedTest.testSynchronizedMethod();
});
thread.start();
}
} public synchronized void testSynchronizedMethod() {
System.out.println("testSynchronizedMethod-start-" + Thread.currentThread().getName());
System.out.println("testSynchronizedMethod-end-" + Thread.currentThread().getName());
}
}

运行结果:

5 个线程中,每个线程有各自的 synchronizedTest 对象。

public class SynchronizedTest {

    public static void main(String[] args) {
for (int i = 0; i < 5; i++) {
SynchronizedTest synchronizedTest = new SynchronizedTest();
Thread thread = new Thread(() -> {
synchronizedTest.testSynchronizedMethod();
});
thread.start();
}
} public synchronized void testSynchronizedMethod() {
System.out.println("testSynchronizedMethod-start-" + Thread.currentThread().getName());
System.out.println("testSynchronizedMethod-end-" + Thread.currentThread().getName());
}
}

运行结果:

我们可以发现如下结论:

  1. 当 5 个线程中只有一个对象时,5 个线程之间的执行顺序是串行的,互不影响,线程0进入方法后,其他线程就无法再进入方法了,得等待线程0执行完后,其他线程才能进入,并且一次只能有1个线程进入。
  2. 每个线程一个对象时,就不会出现排队的场景,各个线程互不影响,相当于每个线程都有各自的资源,没有互相竞争的关系。

由此,我们可以得出什么呢?synchronized 关键字就是悲观锁的具体实现,在多线程并发竞争同一资源时,实现同一时刻只有一个线程能操作资源。

这篇文章是了解 synchronized 最最最皮毛的知识,对 synchronized 熟悉的朋友可以当做是回顾知识点,不熟悉的朋友可以当做是入门,后面会继续深挖知识点。

如果觉得这篇文章看了有收获,麻烦点个在看,支持一下,原创不易。

推荐阅读

写了那么多年 Java 代码,终于 debug 到 JVM 了

全网最新最简单的 openjdk13 代码编译

了解Java线程优先级,更要知道对应操作系统的优先级,不然会踩坑

线程最最基础的知识

老板叫你别阻塞了

吃个快餐都能学到串行、并行、并发

泡一杯茶,学一学同异步

进程知多少?

设计模式看了又忘,忘了又看?

后台回复『设计模式』可以获取《一故事一设计模式》电子书

觉得文章有用帮忙转发&点赞,多谢朋友们!

最新文章

  1. JavaScript:彻底理解同步、异步和事件循环(Event Loop) (转)
  2. cocos2dx中常见设计模式
  3. android NDK 生成so 文件流程-ecplice
  4. c# 纯代码方式创建快捷方式
  5. 十八、ValueStack&ActionContext(非常重点:值栈)
  6. 1234[&#39;toString&#39;][&#39;length&#39;] 等于啥?
  7. Linux mv 命令的10个实用例子
  8. jQuery中的bind绑定事件与文本框改变事件的临时解决方法
  9. 常用ARM指令集及汇编_破解
  10. Angular - - $sce 和 $sceDelegate
  11. 计算机程序的思维逻辑 (74) - 并发容器 - ConcurrentHashMap
  12. 3分钟带你了解PowerShell发展历程——PowerShell各版本资料整理
  13. JBOD磁盘磁盘簇
  14. Android Studio教程10-Intent的详细使用
  15. python双端队列-collection模块
  16. js模块化规范—AMD规范
  17. password、文件MD5加密,passwordsha256、sha384、sha512Hex等加密
  18. Liebig&#39;s Barrels CodeForces - 985C (贪心)
  19. [Algorithm] Radix Sort Algorithm
  20. Java时间的使用

热门文章

  1. HDU - 2444 二分图最大匹配 之 判断二分图+匈牙利算法
  2. React Hooks 实现react-redux
  3. liunx 上安装redis
  4. git本地库中配置多个sshkey
  5. Netty耗时的业务逻辑应该写在哪儿,有什么注意事项?
  6. python ——钟表
  7. Sql练习201908200918
  8. 《深入理解 Java 虚拟机》读书笔记:线程安全与锁优化
  9. P1627 [CQOI2009]中位数 题解
  10. wireshark抓包实战(三),界面菜单管理