Android多线程:这是一份全面&详细的Synchronized学习指南

今天药忘吃喽~ 2022-05-08 07:24 592阅读 0赞

fedcf827170b012c679b233fbf30c8bb.png

前言

  • Java中,有一个常被忽略 但 非常重要的关键字Synchronized
  • 今天,我将详细讲解 Java关键字Synchronized的所有知识,希望你们会喜欢

Carson带你学多线程系列
基础汇总
Android多线程:基础知识汇总
基础使用
Android多线程:继承Thread类使用(含实例教程)
Android多线程:实现Runnable接口使用(含实例教程)
复合使用
Android多线程:AsyncTask使用教程(含实例讲解)
Android多线程:AsyncTask的原理及源码分析
Android多线程:HandlerThread使用教程(含实例讲解)
Android多线程:HandlerThread原理及源码分析
Android多线程:IntentService使用教程(含实例讲解)
Android多线程:IntentService的原理及源码分析
Android多线程:线程池ThreadPool全方位教学
相关使用
Android异步通信:这是一份全面&详细的Handler机制学习攻略
Android多线程:手把手教你全面学习神秘的Synchronized关键字
Android多线程:带你了解神秘的线程变量 ThreadLocal


目录

示意图


1. 定义

Java中的1个关键字


2. 作用

保证同一时刻最多只有1个线程执行 被Synchronized修饰的方法 / 代码

其他线程 必须等待当前线程执行完该方法 / 代码块后才能执行该方法 / 代码块


3. 应用场景

保证线程安全,解决多线程中的并发同步问题(实现的是阻塞型并发),具体场景如下:

  1. 修饰 实例方法 / 代码块时,(同步)保护的是同一个对象方法的调用 & 当前实例对象
  2. 修饰 静态方法 / 代码块时,(同步)保护的是 静态方法的调用 & class 类对象

4. 原理

  1. 依赖 JVM 实现同步
  2. 底层通过一个监视器对象(monitor)完成, wait()notify() 等方法也依赖于 monitor 对象

监视器锁(monitor)的本质 依赖于 底层操作系统的互斥锁(Mutex Lock)实现


5. 具体使用

Synchronized 用于 修饰 代码块、类的实例方法 & 静态方法

5.1 使用规则

示意图

5.2 锁的类型 & 等级

  • 由于Synchronized 会修饰 代码块、类的实例方法 & 静态方法,故分为不同锁的类型
  • 具体如下

示意图

  • 之间的区别

示意图

5.3 使用方式

  1. /**
  2. * 对象锁
  3. */
  4. public class Test{
  5. // 对象锁:形式1(方法锁)
  6. public synchronized void Method1(){
  7. System.out.println("我是对象锁也是方法锁");
  8. try{
  9. Thread.sleep(500);
  10. } catch (InterruptedException e){
  11. e.printStackTrace();
  12. }
  13. }
  14. // 对象锁:形式2(代码块形式)
  15. public void Method2(){
  16. synchronized (this){
  17. System.out.println("我是对象锁");
  18. try{
  19. Thread.sleep(500);
  20. } catch (InterruptedException e){
  21. e.printStackTrace();
  22. }
  23. }
  24. }
  25. /**
  26. * 方法锁(即对象锁中的形式1)
  27. */
  28. public synchronized void Method1(){
  29. System.out.println("我是对象锁也是方法锁");
  30. try{
  31. Thread.sleep(500);
  32. } catch (InterruptedException e){
  33. e.printStackTrace();
  34. }
  35. }
  36. /**
  37. * 类锁
  38. */
  39. public class Test{
  40.    // 类锁:形式1 :锁静态方法
  41. public static synchronized void Method1(){
  42. System.out.println("我是类锁一号");
  43. try{
  44. Thread.sleep(500);
  45. } catch (InterruptedException e){
  46. e.printStackTrace();
  47. }
  48. }
  49. // 类锁:形式2 :锁静态代码块
  50. public void Method2(){
  51. synchronized (Test.class){
  52. System.out.println("我是类锁二号");
  53. try{
  54. Thread.sleep(500);
  55. } catch (InterruptedException e){
  56. e.printStackTrace();
  57. }
  58. }
  59. }

5.4 特别注意

Synchronized修饰方法时存在缺陷:若修饰1个大的方法,将会大大影响效率

  • 示例
    若使用Synchronized关键字修饰 线程类的run(),由于run()在线程的整个生命期内一直在运行,因此将导致它对本类任何Synchronized方法的调用都永远不会成功
  • 解决方案
    使用 Synchronized关键字声明代码块

该解决方案灵活性高:可针对任意代码块 & 任意指定上锁的对象

  1. 代码如下
  2. synchronized(syncObject) {
  3. // 访问或修改被锁保护的共享状态
  4. // 上述方法 必须 获得对象 syncObject(类实例或类)的锁
  5. }

6. 特点

示意图

注:原子性、可见性、有序性的定义

示意图


7. 其他控制并发 / 线程同步方式

7.1 Lock、ReentrantLock

  • 简介

示意图

  • 区别

示意图

7.2 CAS

7.2.1 定义

Compare And Swap,即 比较 并 交换,是一种解决并发操作的乐观锁

synchronized锁住的代码块:同一时刻只能由一个线程访问,属于悲观锁

7.2.2 原理
  1. // CAS的操作参数
  2. 内存位置(A
  3. 预期原值(B
  4. 预期新值(C
  5. // 使用CAS解决并发的原理:
  6. // 1. 首先比较A、B,若相等,则更新A中的值为C、返回True;若不相等,则返回false;
  7. // 2. 通过死循环,以不断尝试尝试更新的方式实现并发
  8. // 伪代码如下
  9. public boolean compareAndSwap(long memoryA, int oldB, int newC){
  10. if(memoryA.get() == oldB){
  11. memoryA.set(newC);
  12. return true;
  13. }
  14. return false;
  15. }
7.2.3 优点

资源耗费少:相对于synchronized,省去了挂起线程、恢复线程的开销

但,若迟迟得不到更新,死循环对CPU资源也是一种浪费

7.2.4 具体实现方式
  • 使用CAS有个“先检查后执行”的操作
  • 而这种操作在Java中是典型的不安全的操作,所以 CAS在实际中是C++通过调用CPU指令实现的
  • 具体过程

    // 1. CAS在Java中的体现为Unsafe类
    // 2. Unsafe类会通过C++直接获取到属性的内存地址
    // 3. 接下来CAS由C++的Atomic::cmpxchg系列方法实现

7.2.5 典型应用:AtomicInteger

对 i++ 与 i–,通过compareAndSet & 一个死循环实现

compareAndSet函数内部 = 通过jni操作CAS指令。直到CAS操作成功跳出循环

  1. private volatile int value;
  2. /**
  3. * Gets the current value.
  4. *
  5. * @return the current value
  6. */
  7. public final int get() {
  8. return value;
  9. }
  10. /**
  11. * Atomically increments by one the current value.
  12. *
  13. * @return the previous value
  14. */
  15. public final int getAndIncrement() {
  16. for (;;) {
  17. int current = get();
  18. int next = current + 1;
  19. if (compareAndSet(current, next))
  20. return current;
  21. }
  22. }
  23. /**
  24. * Atomically decrements by one the current value.
  25. *
  26. * @return the previous value
  27. */
  28. public final int getAndDecrement() {
  29. for (;;) {
  30. int current = get();
  31. int next = current - 1;
  32. if (compareAndSet(current, next))
  33. return current;
  34. }
  35. }

8. 总结

  • 本文主要对Java中常被忽略 但 非常重要的关键字Synchronized进行讲解
  • Carson带你学多线程系列

基础汇总
Android多线程:多线程基础知识汇总
基础使用
Android多线程:继承Thread类使用(含实例教程)
Android多线程:实现Runnable接口使用(含实例教程)
复合使用
Android多线程:AsyncTask使用教程(含实例讲解)
Android多线程:AsyncTask的原理及源码分析
Android多线程:HandlerThread使用教程(含实例讲解)
Android多线程:HandlerThread的原理及源码分析
Android多线程:IntentService使用教程(含实例讲解)
Android多线程:IntentService的原理及源码分析
Android多线程:线程池ThreadPool全方位教学
相关使用
Android异步通信:这是一份全面&详细的Handler机制学习攻略
Android多线程:手把手教你全面学习神秘的Synchronized关键字
Android多线程:带你了解神秘的线程变量 ThreadLocal


欢迎关注Carson_Ho的CSDN博客 与 公众号!

博客链接:https://carsonho.blog.csdn.net/
06b2ed0fbb8979f0aba34c14ede04b59.png


请帮顶 / 评论点赞!因为你的鼓励是我写作的最大动力!

发表评论

表情:
评论列表 (有 0 条评论,592人围观)

还没有评论,来说两句吧...

相关阅读