热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java并发编程之并发编程三大核心问题

个人博客请访问http:www.x0100.top写在前面编写并发程序是比较困难的,因为并发程序极易出现Bug,这些Bug有都是比较诡异的ÿ

个人博客请访问 http://www.x0100.top     

写在前面

编写并发程序是比较困难的,因为并发程序极易出现Bug,这些Bug有都是比较诡异的,很多都是没办法追踪,而且难以复现。

要快速准确的发现并解决这些问题,首先就是要弄清并发编程的本质,并发编程要解决的是什么问题。

本文将带你深入理解并发编程要解决的三大问题:原子性、可见性、有序性。

补充知识

硬件的发展中,一直存在一个矛盾,CPU、内存、I/O设备的速度差异。

速度排序:CPU >> 内存 >> I/O设备

为了平衡这三者的速度差异,做了如下优化:

  1. CPU 增加了缓存,以均衡内存与CPU的速度差异;

  2. 操作系统增加了进程、线程,以分时复用CPU,进而均衡I/O设备与CPU的速度差异;

  3. 编译程序优化指令执行次序,使得缓存能够得到更加合理地利用。


可见性

可见性是什么?

一个线程对共享变量的修改,另外一个线程能够立刻看到,我们称为可见性。

为什么会有可见性问题?

对于如今的多核处理器,每颗CPU都有自己的缓存,而缓存仅仅对它所在的处理器可见,CPU缓存与内存的数据不容易保证一致。

为了避免处理器停顿下来等待向内存写入数据而产生的延迟,处理器使用写缓冲区来临时保存向内存写入的数据。写缓冲区合并对同一内存地址的多次写,并以批处理的方式刷新,也就是说写缓冲区不会即时将数据刷新到主内存中

缓存不能及时刷新导致了可见性问题。

可见性问题举例

public class Test {
public int a = 0;public void increase() {a++;}public static void main(String[] args) {
final Test test = new Test();
for (int i &#61; 0; i <10; i&#43;&#43;) {
new Thread() {
public void run() {
for (int j &#61; 0; j <1000; j&#43;&#43;)test.increase();};}.start();}while (Thread.activeCount() > 1) {
// 保证前面的线程都执行完Thread.yield();}System.out.println(test.a);}
}

目的&#xff1a;10个线程将inc加到10000。

结果&#xff1a;每次运行&#xff0c;得到的结果都小于10000。

原因分析&#xff1a;

假设线程1和线程2同时开始执行&#xff0c;那么第一次都会将a&#61;0 读到各自的CPU缓存里&#xff0c;线程1执行a&#43;&#43;之后a&#61;1&#xff0c;但是此时线程2是看不到线程1中a的值的&#xff0c;所以线程2里a&#61;0&#xff0c;执行a&#43;&#43;后a&#61;1。

线程1和线程2各自CPU缓存里的值都是1&#xff0c;之后线程1和线程2都会将自己缓存中的a&#61;1写入内存&#xff0c;导致内存中a&#61;1&#xff0c;而不是我们期望的2。所以导致最终 a 的值都是小于 10000 的。这就是缓存的可见性问题。

原子性

原子性是什么&#xff1f;

把一个或者多个操作在 CPU 执行的过程中不被中断的特性称为原子性。

在并发编程中&#xff0c;原子性的定义不应该和事务中的原子性&#xff08;一旦代码运行异常可以回滚&#xff09;一样。应该理解为&#xff1a;一段代码&#xff0c;或者一个变量的操作&#xff0c;在一个线程没有执行完之前&#xff0c;不能被其他线程执行。

为什么会有原子性问题&#xff1f;

线程是CPU调度的基本单位。CPU会根据不同的调度算法进行线程调度&#xff0c;将时间片分派给线程。当一个线程获得时间片之后开始执行&#xff0c;在时间片耗尽之后&#xff0c;就会失去CPU使用权。多线程场景下&#xff0c;由于时间片在线程间轮换&#xff0c;就会发生原子性问题

如&#xff1a;对于一段代码&#xff0c;一个线程还没执行完这段代码但是时间片耗尽&#xff0c;在等待CPU分配时间片&#xff0c;此时其他线程可以获取执行这段代码的时间片来执行这段代码&#xff0c;导致多个线程同时执行同一段代码&#xff0c;也就是原子性问题。

线程切换带来原子性问题。

在Java中&#xff0c;对基本数据类型的变量的读取和赋值操作是原子性操作&#xff0c;即这些操作是不可被中断的&#xff0c;要么执行&#xff0c;要么不执行。

i &#61; 0; // 原子性操作
j &#61; i; // 不是原子性操作&#xff0c;包含了两个操作&#xff1a;读取i&#xff0c;将i值赋值给j
i&#43;&#43;; // 不是原子性操作&#xff0c;包含了三个操作&#xff1a;读取i值、i &#43; 1 、将&#43;1结果赋值给i
i &#61; j &#43; 1; // 不是原子性操作&#xff0c;包含了三个操作&#xff1a;读取j值、j &#43; 1 、将&#43;1结果赋值给i

原子性问题举例

还是上文中的代码&#xff0c;10个线程将inc加到10000。假设在保证可见性的情况下&#xff0c;仍然会因为原子性问题导致执行结果达不到预期。为方便看&#xff0c;把代码贴到这里&#xff1a;

public class Test {
public int a &#61; 0;public void increase() {a&#43;&#43;;}public static void main(String[] args) {
final Test test &#61; new Test();
for (int i &#61; 0; i <10; i&#43;&#43;) {
new Thread() {
public void run() {
for (int j &#61; 0; j <1000; j&#43;&#43;)test.increase();};}.start();}while (Thread.activeCount() > 1) {
// 保证前面的线程都执行完Thread.yield();}System.out.println(test.a);}
}

目的&#xff1a;10个线程将inc加到10000。
结果&#xff1a;每次运行&#xff0c;得到的结果都小于10000。

原因分析&#xff1a;

首先来看a&#43;&#43;操作&#xff0c;其实包括三个操作&#xff1a; 

①读取a&#61;0; 

②计算0&#43;1&#61;1; 

③将1赋值给a; 

保证a&#43;&#43;的原子性&#xff0c;就是保证这三个操作在一个线程没有执行完之前&#xff0c;不能被其他线程执行。

实际执行时序图如下&#xff1a;

关键一步&#xff1a;线程2在读取a的值时&#xff0c;线程1还没有完成a&#61;1的赋值操作&#xff0c;导致线程2的计算结果也是a&#61;1。

问题在于没有保证a&#43;&#43;操作的原子性。如果保证a&#43;&#43;的原子性&#xff0c;线程1在执行完三个操作之前&#xff0c;线程2不能执行a&#43;&#43;&#xff0c;那么就可以保证在线程2执行a&#43;&#43;时&#xff0c;读取到a&#61;1&#xff0c;从而得到正确的结果。

有序性

有序性&#xff1a;程序执行的顺序按照代码的先后顺序执行。

编译器为了优化性能&#xff0c;有时候会改变程序中语句的先后顺序。例如程序中&#xff1a;“a&#61;6&#xff1b;b&#61;7&#xff1b;”编译器优化后可能变成“b&#61;7&#xff1b;a&#61;6&#xff1b;”&#xff0c;在这个例子中&#xff0c;编译器调整了语句的顺序&#xff0c;但是不影响程序的最终结果。不过有时候编译器及解释器的优化可能导致意想不到的Bug。

有序性问题举例

Java中的一个经典的案例&#xff1a;利用双重检查创建单例对象

public class Singleton {static Singleton instance;static Singleton getInstance(){if (instance &#61;&#61; null) {synchronized(Singleton.class) {if (instance &#61;&#61; null)instance &#61; new Singleton();}}return instance;}
}

在获取实例getInstance()的方法中&#xff0c;我们首先判断 instance是否为空&#xff0c;如果为空&#xff0c;则锁定 Singleton.class并再次检查instance是否为空&#xff0c;如果还为空则创建Singleton的一个实例。
看似很完美&#xff0c;既保证了线程完全的初始化单例&#xff0c;又经过判断instance为null时再用synchronized同步加锁。但是还有问题&#xff01;

instance &#61; new Singleton(); 创建对象的代码&#xff0c;分为三步&#xff1a;
①分配内存空间
②初始化对象Singleton
③将内存空间的地址赋值给instance

但是这三步经过重排之后&#xff1a;
①分配内存空间
②将内存空间的地址赋值给instance
③初始化对象Singleton

会导致什么结果呢&#xff1f;

线程A先执行getInstance()方法&#xff0c;当执行完指令②时恰好发生了线程切换&#xff0c;切换到了线程B上&#xff1b;如果此时线程B也执行getInstance()方法&#xff0c;那么线程B在执行第一个判断时会发现instance!&#61;null&#xff0c;所以直接返回instance&#xff0c;而此时的instance是没有初始化过的&#xff0c;如果我们这个时候访问instance的成员变量就可能触发空指针异常。

执行时序图&#xff1a;

总结

并发编程的本质就是解决三大问题&#xff1a;原子性、可见性、有序性。

原子性&#xff1a;一个或者多个操作在 CPU 执行的过程中不被中断的特性。由于线程的切换&#xff0c;导致多个线程同时执行同一段代码&#xff0c;带来的原子性问题。

可见性&#xff1a;一个线程对共享变量的修改&#xff0c;另外一个线程能够立刻看到。缓存不能及时刷新导致了可见性问题。

有序性&#xff1a;程序执行的顺序按照代码的先后顺序执行。编译器为了优化性能而改变程序中语句的先后顺序&#xff0c;导致有序性问题。

启发&#xff1a;线程的切换、缓存及编译优化都是为了提高性能&#xff0c;但是引发了并发编程的问题。这也告诉我们技术在解决一个问题时&#xff0c;必然会带来另一个问题&#xff0c;需要我们提前考虑新技术带来的问题以规避风险。


推荐阅读
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文探讨了C语言中指针的应用与价值,指针在C语言中具有灵活性和可变性,通过指针可以操作系统内存和控制外部I/O端口。文章介绍了指针变量和指针的指向变量的含义和用法,以及判断变量数据类型和指向变量或成员变量的类型的方法。还讨论了指针访问数组元素和下标法数组元素的等价关系,以及指针作为函数参数可以改变主调函数变量的值的特点。此外,文章还提到了指针在动态存储分配、链表创建和相关操作中的应用,以及类成员指针与外部变量的区分方法。通过本文的阐述,读者可以更好地理解和应用C语言中的指针。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • t-io 2.0.0发布-法网天眼第一版的回顾和更新说明
    本文回顾了t-io 1.x版本的工程结构和性能数据,并介绍了t-io在码云上的成绩和用户反馈。同时,还提到了@openSeLi同学发布的t-io 30W长连接并发压力测试报告。最后,详细介绍了t-io 2.0.0版本的更新内容,包括更简洁的使用方式和内置的httpsession功能。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 开发笔记:Java是如何读取和写入浏览器Cookies的
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了Java是如何读取和写入浏览器Cookies的相关的知识,希望对你有一定的参考价值。首先我 ... [详细]
  • JDK源码学习之HashTable(附带面试题)的学习笔记
    本文介绍了JDK源码学习之HashTable(附带面试题)的学习笔记,包括HashTable的定义、数据类型、与HashMap的关系和区别。文章提供了干货,并附带了其他相关主题的学习笔记。 ... [详细]
author-avatar
陈初刚5689
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有