Java的并发编程中的多线程问题到底是怎么回事儿?

转载自   Java的并发编程中的多线程问题到底是怎么回事儿?

在我之前的一篇《再有人问你Java内存模型是什么,就把这篇文章发给他。》文章中,介绍了Java内存模型,通过这篇文章,大家应该都知道了Java内存模型的概念以及作用,这篇文章中谈到,在Java并发编程中,通常会遇到三个问题,即原子性问题、一致性问题和有序性问题。

上面一篇文章简单介绍了一下,由于各种原因会导致多线程场景下可能存在原子性、一致性和有序性问题。但是并没有深入,这篇文章就来在之前的基础上,再来看一下,并发编程中,这些问题都是哪来的?

首先,我们还是从操作系统开始,先来了解一些基础知识。

 

CPU时间片

很多人都知道,现在我们用到操作系统,无论是Windows、Linux还是MacOS等其实都是多用户多任务分时操作系统。使用这些操作系统的“用户”是可以“同时”干多件事的,这已经是日常习惯了,并没觉得有什么特别。

但是实际上,对于单CPU的计算机来说,在CPU中,同一时间是只能干一件事儿的。

为了看起来像是“同时干多件事”,分时操作系统是把CPU的时间划分成长短基本相同的时间区间,即”时间片”,通过操作系统的管理,把这些时间片依次轮流地分配给各个“用户”使用。

如果某个“用户”在时间片结束之前,整个任务还没有完成,“用户”就必须进入到就绪状态,放弃CPU,等待下一轮循环。此时CPU又分配给另一个“用户”去使用。

CPU 就好像是一个电话亭,他可以开放给所有用户使用,但是他有规定,每个用户进入电话亭之后只能使用规定时长的时间。如果时间到了,用户还没打完电话,那就会被要求去重新排队。

不同的操作系统,在选择“用户”分配时间片的调度算法是不一样的,常用的有FCFS、轮转、SPN、SRT、HRRN、反馈等,由于不是本文重点,就不展开了。

这个电话亭可以允许哪个用户进入打电话是有不同的策略的,不同的电话亭规定不同,有的电话亭采用排队机制(FCFS)、有的优先分配给打电话时间最短的人(SPN)等

 

进程与线程

前面介绍CPU时间片的时候提到了CPU会根据不同的调度算法把时间片分配给“用户”,这里的“用户”在以前指的是进程,随着操作系统的不断发展,现在一般指线程。

在过去没有线程的操作系统中,资源的分配和执行都是由进程完成的。随着技术的发展,为了减少由于进程切换带来的开销,提升并发能力,操作系统中引入线程。把原本属于进程的工作一分为二,进程还是负责资源的分配,而线程负责执行。

也就是说,进程是资源分配的基本单位,而线程是调度的基本单位。

 

多线程中的并发问题

了解了以上的和硬件及操作系统有关的基础知识以后,我们再来看下,在多线程场景中有哪些并发问题。

关于并发编程中的原子性、可见性和有序性问题我在《内存模型》一文介绍过。

文中提到:缓存一致性问题其实就是可见性问题。而处理器优化是可以导致原子性问题的。指令重排即会导致有序性问题。有部分读者对这部分不是很理解。由于上一篇文章主要介绍内存模型,并没有展开分析,只是给了个结论,这里再针对这部分深入分析一下。

由于缓存一致性问题导致可见性问题,在《内存模型》中介绍的很清晰了,这里就不赘述了,主要结合本文来分析下原子性问题和有序性问题。

原子性问题

我们说原子性问题,其实指的是多线程场景中操作如果不能保证原子性,会导致处理结果和预期不一致。

前面我们提到过,线程是CPU调度的基本单位。CPU有时间片的概念,会根据不同的调度算法进行线程调度。所以在多线程场景下,就会发生原子性问题。因为线程在执行一个读改写操作时,在执行完读改之后,时间片耗完,就会被要求放弃CPU,并等待重新调度。这种情况下,读改写就不是一个原子操作。

就好像我们去电话亭打电话,一共有三个步骤,查找电话,拨号,交流。由于我们在电话亭中可以停留的时间有限,有可能刚刚找到电话号码,时间到了,就被赶出来了。

在单线程中,一个读改写就算不是原子操作也没关系,因为只要这个线程再次被调度,这个操作总是可以执行完的。但是在多线程场景中可能就有问题了。因为多个线程可能会对同一个共享资源进行操作。

比如经典的 i++ 操作,对于一个简单的i++操作,一共有三个步骤:load , add,save 。共享变量就会被多个线程同时进行操作,这样读改写操作就不是原子的,操作完之后共享变量的值会和期望的不一致,举个例子:如果i=1,我们进行两次i++操作,我们期望的结果是3,但是有可能结果是2。

有序性问题

而且,我们知道,除了引入了时间片以外,由于处理器优化和指令重排等,CPU还可能对输入代码进行乱序执行,比如load->add->save 有可能被优化成load->save->add 。这就是有序性问题。

我们打电话的时候,除了可能被中途赶出来以外,本来正常步骤是要查找电话、拨号、交流的。但是电话亭非要给我们优化成查找电话、交流、拨号。这肯定不是我们想要的啊。

还是刚刚的i++操作,在满足了原子性的情况下,如果没有满足有序性,那么得到的结果可能也不是我们想要的。

 

总结

本文主要介绍了并发编程中会导致原子性和有序性问题的原因,关于可见性请参考《内存模型》。关于这三种问题的解决方案在《内存模型》也有介绍,更多的可以参考多线程相关书籍。Hollis后续也会出更多文章再深入分析,敬请期待。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/324893.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DDD理论学习系列(2)-- 领域

1. 引言 领域一词,主要有以下两个意思: 一国主权所达之地。学术思想或社会活动的范围。 不管是指国家的主权范围也好还是学术活动范围,都是在讲一个范围,一个界限。比如我们常说的,学术领域、思想领域、技术领域、语…

洛谷P1462-通往奥格玛瑞的路【日常图论,二分查找,最短路,SPAF】

前言 这道题感谢朋友的帮助,这里是他的博客地址: http://blog.csdn.net/sugar_free_mint 题目 一个无向图,每个点和边都有一定的权值,要求从点1到点2在经过边的权值小于b的情况下经过点的最大权值尽量小 输入 4 4 8(4个点,4条…

《此生未完成》痛句摘录(2)

System.out.print("今天开始继续读书摘录"); //不知道官方让不让我在博客里面记录 //如果不让的话我可能得转到别的上面记录 System.out.print("现在开始看《此生未完成》"); System.out.println("今天是第二天!");他说他已经不再是二…

java正则表达式中的坑String.matches(regex)、Pattern.matches(regex, str)和Matcher.matches()

问题:程序会计算表达式的值 //将数值转换以K为单位 String value "10*1000*1000"; String regex"\\s*\\*\\s*1000\\s*"; boolean isMatch value .matches(regex); if(isMatch){value value.replaceFirst(regex,""); }else{String…

最全、最详细的配置jdk十步法!

求关注求转发大家好,我是雄雄。今天给大家分享的是 如何配置jdk ,还记得我当年刚开始自学编程时,jdk配置了好几天才配置好,说来也是惭愧啊。并且配置成功的步骤我自己都不知道是啥,因为当时是东找一个方法,…

2018GDKOI——记录

前言 诶呀,这次GDKOI让我失去了AK的梦想 正题 感想 诶呀,题目真是难。反正不会,听课依旧半点不懂,所以我相信“暴力出奇迹”。 然后水总分120,两天都水得了60。 day1 题1 题目: 就是一个模拟地铁的&a…

interface关键字

接口的使用 1.接口使用interface来定义 2.Java中,接口和类是并列的两个结构 3.如何定义接口:定义接口中的成员3.1 JDK7及以前:只能定义全局常量和抽象方法>全局常量:public static final的.但是书写时,可以省略不写…

RabbitMQ系列教程之三:发布\/订阅(Publish\/Subscribe)

在前一个教程中,我们创建了一个工作队列。工作队列背后的假设是每个任务会被交付给一个【工人】。在这一部分我们将做一些完全不同的事情--我们将向多个【消费者】传递信息。这种模式被称为“发布/订阅”。 为了说明这种模式,我们将构建一个简单的日志…

路径使用场景

路径的使用场景 路径适用的范围:jsp页面中link script img a form 等 如果访问的是其他服务器中的项目文件只可以使用绝对路径 如果访问的是同一个服务器中的其他项目推荐使用根路径 如果访问的是同一个服务器中的同一个项目中的文件推荐使用相对base标签的路径 修改…

《此生未完成》痛句摘抄(3)

System.out.print("今天开始继续读书摘录"); //不知道官方让不让我在博客里面记录 //如果不让的话我可能得转到别的上面记录 System.out.print("现在开始看《此生未完成》"); System.out.println("今天是第三天!");有时候常常会想到那…

【jzoj】2018.1.30NOIP普及组——模拟赛D组

_ 前言 写博客时间 2018/1/30 22:36。 感想:出数据的dalao我服! 正题 题目1:二项式展开式(jzoj2254) 输入一个整数,求展开(ab)^n。展开方式为 (ab)^n?a^n?a^(n-1)b?a^(n-2)b^2…?b^n 其中” ? “…

开发人员也要懂点的测试知识

转载自 开发人员也要懂点的测试知识 本文来自于作者投稿,作者陈彩华,贝聊后端开发工程师。 最近参加了保利威测试总监李乐的《互联网测试姿势》为主题的分享交流会,收获颇丰,作为一个开放,秉承“不懂产品和测试的开…

Java8中接口的新特性

知识点1: 接口中定义的静态方法,只能通过接口来调用。 CompareA.method1(); 知识点2: 通过实现类的对象,可以调用接口中的默认方法。如果实现类重写了接口中的默认方法,调用时,仍然调用的是重写以后的方…

转发和重定向

需求说明 用户输入并提交登录信息 登录成功,跳转到success.jsp页面 登录失败,跳转回login.jsp页面 理解图 区别详解

实现自己的.NET Core配置Provider之Yaml

YAML是一种更适合人阅读的文件格式,很多大型的项目像Ruby on Rails都选择YAML作为配置文件的格式。如果项目的配置很少,用JSON或YAML没有多大差别。看看rails项目中的配置文件,如果用JSON写试试什么感受吧。 在《实现自己的.NET Core配置Pro…

信息时代与人工智能时代的教育变革

求关注求转发本文是雄雄的小课堂投稿的第 1 篇文章,作者:小溪时代发展迅速,学校的教育体系却没有很大的改变。比如现在的学生们还是学着几百年前的东西。试想一下,这是否真的适合我们这个时代,亦或真的需要改变了。教育…

【jzoj】2018.1.31 NOIP普及组——D组模拟赛

前言 今天题目比较水and我进了C组,不过太太太太太太太太太太太太太太太太绝望了QAQ。所以我也没有做C组的题。写完博客我就做O(∩_∩)O。 正题 题1:奇数统计(jzoj1547) 就是输入n个数,输出出现次数为奇数的一个数&a…

请求和响应向更多内容

1、请求更多方法 /*******[1]获得req对象中的头信息*(了解)*******************/Enumeration<String> headerNames req.getHeaderNames();while(headerNames.hasMoreElements()){String next headerNames.nextElement();System.out.println(next"-----"req.…

大数据Big Data

转载自 大数据Big Data 2012年本站曾对大数据预测&#xff1a;如果说2012年是大数据概念为人所知、引人瞩目、小试牛刀的一年&#xff0c;那么2013年大数据将会实现产品部署&#xff0c;早期投资获得回报&#xff0c;一小部分的产业被颠覆。到了2014年&#xff0c;各种大数据项…

RabbitMQ系列教程之四:路由(Routing)

在上一个教程中&#xff0c;我们构建了一个简单的日志系统&#xff0c;我们能够向许多消息接受者广播发送日志消息。在本教程中&#xff0c;我们将为其添加一项功能 &#xff0c;这个功能是我们将只订阅消息的一个子集成为可能。 例如&#xff0c;我们可以只将关键的错误消息输…