分析Java AtomicInteger

简介: 为了解决多线程访问Integer变量导致结果不正确所设计的一个基于多线程并且支持原子操作的Integer类AtomicInteger内部有一个变量UnSafe:Unsafe类是一个可以执行不安全、容易犯错的操作的一个特殊类虽然Unsafe类中所有方法都是public的,但是这个类只能在一些被信任的代码中使用。

为了解决多线程访问Integer变量导致结果不正确所设计的一个基于多线程并且支持原子操作的Integer类

AtomicInteger内部有一个变量UnSafe:


img_b5bf97b89cdff77eec4b1174e9bbdc03.png

Unsafe类是一个可以执行不安全、容易犯错的操作的一个特殊类
虽然Unsafe类中所有方法都是public的,但是这个类只能在一些被信任的代码中使用。

Unsafe类可以执行以下几种操作

  • 分配内存,释放内存:在方法allocateMemory,reallocateMemory,freeMemory中,有点类似c中的malloc,free方法
  • 可以定位对象的属性在内存中的位置,可以修改对象的属性值。使用objectFieldOffset方法
  • 挂起和恢复线程,被封装在LockSupport类中供使用

AtomicInteger中用的就是Unsafe的CAS操作
Unsafe中的int类型的CAS操作方法

public final native boolean compareAndSwapInt(Object o, long offset,int expected,int x);

如果我把1这个数字属性更新到2的话,需要这样调用:

compareAndSwapInt(this, valueOffset, 1, 2)

valueOffset字段表示内存位置,可以在AtomicInteger对象中使用unsafe得到:


img_2b4b116e6b42b5c52609c8e3852ecefc.png
`

value表示当前的整型值,这个整型变量还是volatile的,保证内存可见


img_6fdc2c53718ebc654621eb4e25db3264.png

AI内部还封装了一下CAS,定义了一个compareAndSet方法,只需要2个参数


img_b24cedbb4ed9ec4406090381096476b8.png

addAndGet方法

内部使用一个死循环,先得到当前的值value,然后再把当前的值加一,加完之后使用cas原子操作让当前值加一处理正确。当然cas原子操作不一定是成功的,所以做了一个死循环,当cas操作成功的时候返回数据。这里由于使用了cas原子操作,所以不会出现多线程处理错误的问题。

比如线程A得到current为1,线程B也得到current为1
线程A的next值为2,进行cas操作并且成功的时候,将value修改成了2
这个时候线程B也得到next值为2,当进行cas操作的时候由于expected值已经是2,而不是1了
所以cas操作会失败,下一次循环的时候得到的current就变成了2
也就不会出现多线程处理问题了


img_ceb8efbf639d0c5989f9a5e4448eac4e.png

img_06ddcefb141cb00b54ff8440d032a48e.png

incrementAndGet方法

跟addAndGet方法类似,只不过next值变成了current+1:


img_4264eae79207f88b601116428ab21811.png

getAndAdd方法

跟addAndGet方法一样,返回值变成了current:


img_3da3ab09deae66932856509a4e27dd03.png

缺点

虽然AtomicInteger中的cas操作可以实现非阻塞的原子操作,但是会产生ABA问题,

目录
相关文章
|
3月前
|
存储 Java
【编程基础知识】 分析学生成绩:用Java二维数组存储与输出
本文介绍如何使用Java二维数组存储和处理多个学生的各科成绩,包括成绩的输入、存储及格式化输出,适合初学者实践Java基础知识。
101 1
|
17天前
|
缓存 算法 搜索推荐
Java中的算法优化与复杂度分析
在Java开发中,理解和优化算法的时间复杂度和空间复杂度是提升程序性能的关键。通过合理选择数据结构、避免重复计算、应用分治法等策略,可以显著提高算法效率。在实际开发中,应该根据具体需求和场景,选择合适的优化方法,从而编写出高效、可靠的代码。
27 6
|
2月前
|
监控 算法 Java
jvm-48-java 变更导致压测应用性能下降,如何分析定位原因?
【11月更文挑战第17天】当JVM相关变更导致压测应用性能下降时,可通过检查变更内容(如JVM参数、Java版本、代码变更)、收集性能监控数据(使用JVM监控工具、应用性能监控工具、系统资源监控)、分析垃圾回收情况(GC日志分析、内存泄漏检查)、分析线程和锁(线程状态分析、锁竞争分析)及分析代码执行路径(使用代码性能分析工具、代码审查)等步骤来定位和解决问题。
|
2月前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
73 2
|
2月前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
47 2
|
3月前
|
Java
让星星⭐月亮告诉你,Java synchronized(*.class) synchronized 方法 synchronized(this)分析
本文通过Java代码示例,介绍了`synchronized`关键字在类和实例方法上的使用。总结了三种情况:1) 类级别的锁,多个实例对象在同一时刻只能有一个获取锁;2) 实例方法级别的锁,多个实例对象可以同时执行;3) 同一实例对象的多个线程,同一时刻只能有一个线程执行同步方法。
26 1
|
3月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
63 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
3月前
|
Java
如何从Java字节码角度分析问题|8月更文挑战
如何从Java字节码角度分析问题|8月更文挑战
|
3月前
|
安全 网络协议 Java
Java反序列化漏洞与URLDNS利用链分析
Java反序列化漏洞与URLDNS利用链分析
75 3
|
3月前
|
存储 Java 编译器
[Java]基本数据类型与引用类型赋值的底层分析
本文详细分析了Java中不同类型引用的存储方式,包括int、Integer、int[]、Integer[]等,并探讨了byte与其他类型间的转换及String的相关特性。文章通过多个示例解释了引用和对象的存储位置,以及字符串常量池的使用。此外,还对比了String和StringBuilder的性能差异,帮助读者深入理解Java内存管理机制。
33 0