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

线程安全问题及解决

1.前言

当我们使用多个线程访问同一资源时(可以是同一变量,同一文件,同一条记录),若多个线程只要只读操作,则不会发生线程安全问题;如果多个线程既有可读又有可写操作时,将可能导致线程安全问题.

2.提出问题

例 : 三个人对银行账户存储的100块存款进行取钱,如果该账户还有存款,就可以取.该问题可能发生线程安全问题吗?

3.继承Thread类的方式进行模拟 : 

public class ThreadTest {public static void main(String[] args) {MulterThread t1 = new MulterThread("线程-1");MulterThread t2 = new MulterThread("线程-2");MulterThread t3 = new MulterThread("线程-3");t1.start();t2.start();t3.start();}
}
class MulterThread extends Thread {static int change = 100;public MulterThread() {super();}public MulterThread(String name) {super(name);}@Overridepublic void run() {while(true) {if (change > 0) {System.out.println(Thread.currentThread().getName() + "\t\t" + change);change--;} else {break;}}}
}控制台 : 
//显然有问题,100的时候被取的两次
线程-2		100
线程-1		100
线程-2		99
线程-1		98
线程-2		97
线程-1		96
线程-2		95
线程-1		94
线程-1		92
线程-2		93
略

注 : 

  • 为什么change变量要声明为static : 如果不声明为static,那么new了三个MulterThread对象,就会有300块的存款,与抢占同一资源的场景不符.
  • 为什么会出现两次100呢 : 很显然,每次运行结果不一样,按该次运行结果举例.当线程2调用run()方法进入输出语句的时候,执行到下一句change--还需要一段时间,而此时线程1也调用了run(),并也执行到了输出语句,此时change--语句并未执行,所以二者都打印的是100.

3.实现Runnable接口的方法进行模拟

public class RunnableTest {public static void main(String[] args) {A a = new A();Thread t1 = new Thread(a);Thread t2 = new Thread(a);Thread t3 = new Thread(a);t1.start();t2.start();t3.start();}}
class A implements Runnable{int change = 100;@Overridepublic void run() {while (true) {if (change > 0) {System.out.println(Thread.currentThread().getName() + "\t\t" + change);change--;} else {break;}}}
}控制台 : 
Thread-1		100
Thread-1		99
Thread-1		98
Thread-0		100
Thread-1		97
Thread-0		96
Thread-2		100
Thread-0		94
Thread-1		95
Thread-1		91
略

注 : 

  • 为什么change变量不用static修饰 : 只调用一次new创建了A的一个对象,并作为同一个实参传入到Thread类中.因为只new了一次,所以change只有一份.
  • 为什么会出现三次100 : 与上同.

4.解决方案

必须满足一个线程在操作change时,其他线程必须等待,直到该线程操作完成后,其他线程才可以进来操作change.

5.方式1 : 同步代码块

(1). 格式

synchronized(同步监视器){

    //需要被同步的代码

}

(2). 利用锁来解决继承Thread类带来的线程安全问题.

@Overridepublic void run() {while(true) {synchronized (MulterThread.class){if (change > 0) {System.out.println(Thread.currentThread().getName() + "\t\t" + change);change--;} else {break;}}}}

(3). 利用锁来解决实现接口带来的线程安全问题 : 

@Overridepublic void run() {while (true) {synchronized (this){if (change > 0) {System.out.println(Thread.currentThread().getName() + "\t\t" + change);change--;} else {break;}}}}

说明 : 

  • 需要被同步的代码,即为操作共享数据的代码.
  • 共享数据 : 即多个线程可以操作的数据 : 如该处的change.
  • 需要被同步的代码,在被synchronized包裹后,就使得一个线程操作共享数据时,其他线程需等待.
  • 同步监视器(锁) : 哪个线程获得了锁,哪个线程就可以执行被同步的代码.
  • 锁可以由任何对象充当,但必须多个线程共用同一个同步监视器.(即该监视器必须唯一).
  • 继承Thread类 : 锁---->类名.class
  • 实现接口 : 锁------>this

相关文章:

  • mysql--事务四大特性与隔离级别
  • Neo4j桌面版导入CVS文件
  • 利用瑞士军刀netcat建立连接并实现文件上传
  • 从姿态估计到3D动画
  • 1.7.2 python练习题15道
  • C++超市商品管理系统
  • 计算机网络基础——网络安全/ 网络通信介质
  • 大学 Python 程序设计实验报告:判断变量名是否合法
  • XSKY 智能存储,助力“数据要素 X”先进制造
  • openGauss Index-advisor_索引推荐
  • docker 的八大技术架构(图解)
  • 【Web前端】CSS基本语法规范和引入方式常见选择器用法常见元素属性
  • Android 观察者模式
  • 【KMeans聚类概述】
  • HTTPS握手解析
  • 4个实用的微服务测试策略
  • angular2开源库收集
  • GitUp, 你不可错过的秀外慧中的git工具
  • java8-模拟hadoop
  • Laravel深入学习6 - 应用体系结构:解耦事件处理器
  • maven工程打包jar以及java jar命令的classpath使用
  • node入门
  • 笨办法学C 练习34:动态数组
  • 第十八天-企业应用架构模式-基本模式
  • 分享几个不错的工具
  • 搞机器学习要哪些技能
  • 浅谈JavaScript的面向对象和它的封装、继承、多态
  • 数据结构java版之冒泡排序及优化
  • 用mpvue开发微信小程序
  • 终端用户监控:真实用户监控还是模拟监控?
  • ​创新驱动,边缘计算领袖:亚马逊云科技海外服务器服务再进化
  • ​力扣解法汇总1802. 有界数组中指定下标处的最大值
  • $.each()与$(selector).each()
  • (01)ORB-SLAM2源码无死角解析-(66) BA优化(g2o)→闭环线程:Optimizer::GlobalBundleAdjustemnt→全局优化
  • (3)STL算法之搜索
  • (poj1.3.2)1791(构造法模拟)
  • (超简单)构建高可用网络应用:使用Nginx进行负载均衡与健康检查
  • (二)Eureka服务搭建,服务注册,服务发现
  • (黑马C++)L06 重载与继承
  • (考研湖科大教书匠计算机网络)第一章概述-第五节1:计算机网络体系结构之分层思想和举例
  • (力扣)1314.矩阵区域和
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第3章 信息系统治理(一)
  • (牛客腾讯思维编程题)编码编码分组打印下标题目分析
  • (四)鸿鹄云架构一服务注册中心
  • (心得)获取一个数二进制序列中所有的偶数位和奇数位, 分别输出二进制序列。
  • (一)使用Mybatis实现在student数据库中插入一个学生信息
  • (一)硬件制作--从零开始自制linux掌上电脑(F1C200S) <嵌入式项目>
  • (转) ns2/nam与nam实现相关的文件
  • (转)JAVA中的堆栈
  • (转)Windows2003安全设置/维护
  • (转载)PyTorch代码规范最佳实践和样式指南
  • *++p:p先自+,然后*p,最终为3 ++*p:先*p,即arr[0]=1,然后再++,最终为2 *p++:值为arr[0],即1,该语句执行完毕后,p指向arr[1]
  • .bat批处理(三):变量声明、设置、拼接、截取
  • .class文件转换.java_从一个class文件深入理解Java字节码结构
  • .NET 使用 ILMerge 合并多个程序集,避免引入额外的依赖