当前位置: 首页 > news >正文

【高级编程】synchronized 解决并发问题 类的线程安全类型

文章目录

    • 并发问题
      • 同步方法
      • 同步代码块
    • 线程安全类型
      • ArrayList
      • Hashtable
      • HashMap
      • Vector

多线程共享数据引发的问题

模拟 “A” “B” “C” 三人抢票,总票数10张,打印抢票情况以及剩余票数。

public class Site implements Runnable {int count = 10; // 总票数int num = 0;    // 已抢票数@Overridepublic void run() {while (true) {if (count <= 0) {break;}count--;num++;System.out.println(Thread.currentThread().getName() + "抢到第" + num + "张票,剩余" + count + "张票!");//休眠  模拟网络延时try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();
}}}}
public static void main(String[] args) {Site site = new Site();Thread th1 = new Thread(site,"A");Thread th2 = new Thread(site,"B");Thread th3 = new Thread(site,"C");th1.start();th2.start();th3.start();
}

问题

  • 不是从第1张票开始

  • 存在多人抢到一张票的情况

  • 有些票号没有被抢到

多个线程操作同一共享资源时,将引发数据不安全问题

并发问题

synchronized 在 Java 中是一种悲观锁(Pessimistic Lock)的形式。悲观锁假设最坏的情况,即认为数据总是会被其他线程修改,因此在处理数据之前就先获取锁。这样可以避免数据冲突,但也可能导致较高的竞争开销。

同步方法

使用 synchronized 修饰的方法控制对类成员变量的访问,synchronized 就是为当前的线程声明一把锁

访问修饰符 synchronized 返回类型 方法名(参数列表){……}

synchronized 访问修饰符 返回类型 方法名(参数列表){……}

使用同步方法的网络购票

public class Site implements Runnable{int count = 10; // 总票数int num = 0;    // 已抢票数@Overridepublic void run() {while(true){if(qg()){ break; }//休眠  模拟网络延时try {Thread.sleep(500); // 休眠半秒} catch (InterruptedException e) {e.printStackTrace();}}}public synchronized boolean qg(){if(count <= 0){ return true; }count--;num++;System.out.println(Thread.currentThread().getName()+"抢到第"+num+"张票,剩余"+count+"张票!");return false;}}

同步代码块

使用 synchronized 关键字修饰的代码块

synchronized(syncObject){// 需要同步的代码
}
  • syncObject 为需同步的对象,通常为 this
  • 效果与同步方法相同

使用同步代码块的网络购票

public class Site implements Runnable{int count = 10; // 总票数int num = 0;    // 已抢票数@Overridepublic void run() {while(true){//同步synchronized (this){if(count <= 0){ break;}count--;num++;System.out.println(Thread.currentThread().getName()+"抢到第"+num+"张票,剩余"+count+"张票!");}//休眠  模拟网络延时try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();
}}}}

多个并发线程访问同一资源的同步代码块时

  • 同一时刻只能有一个线程进入同步代码块

  • 当一个线程访问一个同步代码块时,其他同步代码块同样被锁定

  • 当一个线程访问一个同步代码块时,其他线程可以访问该资源的非同步代码

线程安全类型

为达到安全性和效率的平衡,可以根据实际场景来选择合适的类型

方法是否同步效率比较适合场景
线程安全多线程并发共享资源
非线程安全单线程
HashtableHashMap
继承关系实现了 Map 接口,继承 Dictionary实现了 Map 接口,继承 AbstractMap
安全性线程安全,效率较低非线程安全,效率较高
键值键和值都不允许为 null键和值都允许为 null
StringBufferStringBuilder
安全性线程安全非线程安全
适合场景适用于多线程环境中的字符串大量操作适用于单线程环境中的字符串拼接

ArrayList

非线程安全

// 查看 ArrayList 类的 add() 方法定义
public boolean add(E e) {ensureCapacityInternal(size + 1); // 集合扩容,确保能新增数据elementData[size++] = e; // 在新增位置存放数据return true;
}

ArrayList 类的 add() 方法为非同步方法。当多个线程向同一个 ArrayList 对象添加数据时,可能出现数据不一致问题

Hashtable

线程安全

Hashtable<String,String> hashtable = new Hashtable<>();
hashtable.put("","");public synchronized V put(K key, V value) {// Make sure the value is not nullif (value == null) {throw new NullPointerException();}// Makes sure the key is not already in the hashtable.......
}

HashMap

非线程安全

HashMap:在多线程环境中,可能会导致数据丢失或结构破坏。

HashMap<String,String> hashMap = new HashMap<>();
hashMap.put("","");public V put(K key, V value) {return putVal(hash(key), key, value, false, true);
}

Vector

线程安全

Vector vector = new Vector();
vector.add("");public synchronized boolean add(E e) {modCount++;ensureCapacityHelper(elementCount + 1);elementData[elementCount++] = e;return true;
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • wireshark打开时空白|没有接口,卸载重装可以解决
  • iOS的传递链与响应链机制
  • CSP-J算法基础 树状结构与二叉树
  • 学习笔记 - 知识图谱的符号表示方法
  • C#中的装箱和拆箱是什么
  • Sentinel 控制界面
  • element form rules 验证数组对象属性时如何写判断规则
  • 组合总和IV(力扣---动态规划)
  • 多态(上)【C++】
  • 如何搞定日语翻译?试试这四款工具
  • FreeRTOS学习(2)延时函数的封装
  • 【白话树】之 树的基本知识、存储结构和二叉树转换
  • MySQL 子查询
  • hiresfix_latent 使用Upscale放大节点 对图片进行放大Comfyui
  • 众店绿色积分模式:引领消费新风尚,共筑商业新生态
  • [笔记] php常见简单功能及函数
  • 《深入 React 技术栈》
  • 002-读书笔记-JavaScript高级程序设计 在HTML中使用JavaScript
  • Angular 响应式表单 基础例子
  • ES6系统学习----从Apollo Client看解构赋值
  • JavaScript 奇技淫巧
  • java小心机(3)| 浅析finalize()
  • js学习笔记
  • Redis中的lru算法实现
  • swift基础之_对象 实例方法 对象方法。
  • 阿里云前端周刊 - 第 26 期
  • 基于 Ueditor 的现代化编辑器 Neditor 1.5.4 发布
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • 将 Measurements 和 Units 应用到物理学
  • 讲清楚之javascript作用域
  • 前端js -- this指向总结。
  • 前端设计模式
  • 微信公众号开发小记——5.python微信红包
  • Hibernate主键生成策略及选择
  • 不要一棍子打翻所有黑盒模型,其实可以让它们发挥作用 ...
  • ​什么是bug?bug的源头在哪里?
  • ​一帧图像的Android之旅 :应用的首个绘制请求
  • #、%和$符号在OGNL表达式中经常出现
  • #1015 : KMP算法
  • #java学习笔记(面向对象)----(未完结)
  • (003)SlickEdit Unity的补全
  • (32位汇编 五)mov/add/sub/and/or/xor/not
  • (delphi11最新学习资料) Object Pascal 学习笔记---第7章第3节(封装和窗体)
  • (iPhone/iPad开发)在UIWebView中自定义菜单栏
  • (Oracle)SQL优化技巧(一):分页查询
  • (二)hibernate配置管理
  • (附源码)springboot教学评价 毕业设计 641310
  • (附源码)ssm捐赠救助系统 毕业设计 060945
  • (附源码)ssm旅游企业财务管理系统 毕业设计 102100
  • (十六)视图变换 正交投影 透视投影
  • (四)事件系统
  • (原)本想说脏话,奈何已放下
  • .DFS.
  • .NET 5.0正式发布,有什么功能特性(翻译)
  • .NET BackgroundWorker