超越线程池:java并发并没有你想的那么糟糕-mile米乐体育
很多人一直唠叨着并发中的新概念。然而,许多开发人员还没有机会把过多的注意力都放在上面。在这篇文章中,我们将带您了解java 8 streams、 hadoop、 apache spark、 quasar fibers以及响应式编程,让你迅速入门。尤其是如果你不经常用它们的话。一句话,它并不遥远,它就在我们身边。
谈到并发,一种很好的方式来形容当前的问题是来回答几个小问题以便更好的了解它:
它是一个数据处理任务么?如果是这样的话,它可以分解为独立的任务单元么?
操作系统、虚拟机和你的代码之间的关系是什么?(本地线程 vs 轻量级线程)
有多少机器和处理器参与?(单核 vs 多核)
让我们带着问题,一起找出每个问题的最佳答案吧。
1、从线程池到并行流
在java 8中,我们了解到新的流api接口,它允许应用聚集操作,如筛选、排序或者映射数据流。流允许我们做的另一件事情是,在多核机器上应用并行操作。并行流 ——通过把fork/join框架引入java 7将线程间的工作分离。java 6并发库,我们看到了executorservice创建和处理我们的工作线程池,这不得不说是个进步。
fork/join也建立在executorservice之上,与传统的线程主要的区别在于如何在线程和支持多核的机器间分配工作。用一个简单的 executorservice你能完全控制工作线程之间的负载分布,确立每个任务的大小以便线程来处理。而fork/join,恰好有个work-stealing算法分配线程间的负载。简而言之,这允许大型任务可以被分成更小单元,并在不同的线程间处理,最终我们可以知道——它是为了平衡线程间的 工作。然而,这并不是万能的。
有时并行流会减慢你速度的,所以你需要多想想。在你的方法中使用parallelstream会导致瓶颈和减速(在我们基准测试中跑慢了约15%左右)。假设我们已经运行多个线程,在其中一些我们使用parallelstream,在线程池中添加越来越多的线程。这可以很容易超过我们的核心处理能力,由于增加了上下文转换一切都慢下来了。
小结:在单机上并行流使线程处理抽象化,在一定程度上这会均衡核心间的负载。然而,如果你想高效使用它们,记住硬件是关键而不是生产更多的线程而超出机器的处理能力。
2、apache hadoop和apache spark
接下来谈多核机器、 pb级数据和任务,这跟所有从twitter提到的java或重载机器学习算法类似。谈到hadoop,不得不说这个应用广泛的框架及它的组 件:hadoop分布式文件系统(hdfs)、资源管理平台(yarn)、数据处理模块(mapreduce)和其他所需的类库和工具(common)。 在这些组件上层还有一些其他很受欢迎的可选工具,比如运行在hdfs上的数据库(hbase)、查询语言平台(pig)和数据仓库基础结构(hive)。
apache spark 作为一种新数据处理模块,以内存性能和快速执行的弹性分布式数据集(rdds)而出名,不同于不能高效使用内存和磁盘的hadoop mapreduce。databricks公布的最新标准显示当用少于10倍节点的时候,对1pb数据的排序spark比hadoop快三倍。
典型的hadoop用例在于查询数据,而spark正以其快速的机器学习算法越来越出名。但这只是冰山一角,databricks如是说:“spark 使应用程序在hadoop集群中运行在内存中快100倍,当运行在磁盘中时甚至快10倍”。
小结:spark是在hadoop生态系统中的后起之秀,有一个常见的误解是我们现在经常谈它一些不合作或竞争的事情,但是我认为我们在这正在看到这个框架的发展。
3、quasar fibers
我们有机会运行在hadoop,现在让我们回到单机。事实上,在java多线程应用程序和集中在单线程上,让我们眼光再长远些。就我们而言,hotspot jvm线程与本地系统线程相同,持有一个线程并且运行在”虚拟“线程中,这在fibers中都包含的。java没有原生的fibers支持,但是不要担 心,quasar通过parallel universe解决了我们的问题。
quasar 是一个开源的jvm库。它支持fibers(也称为轻量级线程),并且还充当框架的角色,在后面中我会提到。在这上下文转换是它本质的名字。当我们核心数 量有限,一旦本地线程数量越大我们就会收到越来越多的上下文开销。一种解决这个问题的方式是fibers,使用单线程支持”多线程“。这看起来像threadcepiton的一个实例。
fibers还可以被视为一个从线程池的进化,当我们通过应用并行流的时候避开了线程过载的危险。他们更容易衡量线程和允许令人可观的并行”轻量“线程数量。它们不是为了取代线程,而是应该用在那些相对来说经常堵塞的代码中,就如同担任真正异步线程的角色。
小结:并行领域在java并发性中正提供一种新的思路,虽然还没有版本发布,但是值得一试。
4、actor和响应式编程
在响应式的官方言论中,最新的释义有4原则:响应、有弹性、灵活性和消息驱动。这基本意味着快速、容错、可伸缩的和支持非阻塞通信。
让我们看看akka actor是如果支持它的吧。简单来讲actor有一个状态和一个特定的行为,通过交换消息沟通彼此的邮箱。一个actor系统作为一个整体应该被每个应 用程序创建,拥有一个层次结构将任务分解成更小的任务以便每个角色最多只有一个监督的角色。一个角色也可以处理这个任务,通过委托给另一个角色将其进一步 分解或在实例失败的情况下,将它反馈给它的监督者。无论哪种方式,消息不应该包括行为或者共享可变的状态,每个角色都有一个独立的状态和行为。
它是一个从大多数开发者在使用的并发模型的思考模式的转移。尽管它起源于70年代,但是为了适应现代应用程序的要求,直到最近几年它才复苏。并行领域的quasar也支持actor,实现的主要区别在于fibers/轻量级线程。
小结:相反的,actor模型需要管理线程池,让它远离使用工具包。今天面对这种应用程序处理的问题,尤其在我们可以处理拥有更多核心的高并发系统方面又重新有了关注。
关于使用并发或者并行算法,我们今天通过介绍4种方法来解决问题来应对你需要的场景。希望这有 助于激起你的兴趣,以及在这大谈并发话题的现在开拓下你的视野。超越线程池,有一种将这委托给语言及它的工具的趋势——关注新的技术并应用它而不是花费无 数个小时解决竞态条件和锁。