织机原理_项目织机

织机原理

为什么为什么?

Java 8流背后的驱动程序之一是并发编程。 在流管道中,指定要完成的工作,然后任务将自动分配到可用处理器上:

var result = myData.parallelStream().map(someBusyOperation).reduce(someAssociativeBinOp).orElse(someDefault);

当数据结构便宜且可拆分成多个部分且操作使处理器繁忙时,并行流将发挥出色的作用。 这就是它的设计目的。

但是,如果您的工作负载包含大部分阻塞的任务,那么这对您没有帮助。 那是您典型的Web应用程序,可以处理许多请求,每个请求都花费大量时间等待REST服务,数据库查询等结果。

1998年,令人惊奇的是,Sun Java Web Server(Tomcat的前身)在单独的线程而不是OS进程中运行每个请求。 这样就可以满足数千个并发请求! 如今,这并不令人惊讶。 每个线程占用大量内存,典型服务器上不能有数百万个线程。

这就是为什么服务器端编程的现代口号是:“永不阻塞!” 相反,您指定一旦数据可用就应该发生什么。

这种异步编程风格非常适合服务器,使它们可以轻松支持数百万个并发请求。 对于程序员来说不是那么好。

这是使用HttpClient API的异步请求:

HttpClient.newBuilder().build().sendAsync(request, HttpResponse.BodyHandlers.ofString()).thenAccept(response -> . . .);.thenApply(. . .);.exceptionally(. . .);

我们通常用语句实现的功能现在被编码为方法调用。 如果我们喜欢这种编程风格,就不会在Lisp中使用我们的编程语言来编写语句和编写快乐的代码。

诸如JavaScript和Kotlin之类的语言为我们提供了“异步”方法,在该方法中,我们编写语句,然后将这些语句转换为您刚刚看到的方法调用。 很好,只不过它意味着现在有两种方法-常规方法和转换方法。 而且您不能混合使用它们(“红色药丸/蓝色药丸”分界)。

Project Loom从Erlang和Go等语言中获得指导,在这些语言中,阻塞并不是什么大问题。 您可以在“光纤”或“轻型线程”或“虚拟线程”中运行任务。 该名称尚待讨论,但我更喜欢“光纤”,因为它很好地表示了多个光纤在一个承载线程中执行的事实。 当发生阻塞操作(例如等待锁定或I / O)时,光纤将停放。 停车比较便宜。 如果很多时候都停放了一根载运纤维,则可以支撑一千根纤维。

请记住,Project Loom不能解决所有并发问题。 如果您有大量计算任务,并且想让所有处理器内核都忙,它对您无济于事。 它对使用单个线程的用户界面没有帮助(用于序列化对不是线程安全的数据结构的访问)。 在该用例中继续使用AsyncTask / SwingWorker / JavaFX Task 。 当您有很多任务花费大量时间阻塞时,Project Loom很有用。

注意 如果您已经待了很长时间,您可能还记得Java的早期版本具有映射到OS线程的“绿色线程”。 但是,有一个关键的区别。 当绿色线程被阻塞时,其承载线程也被阻塞,从而阻止了同一承载线程上的所有其他绿色线程取得进展。

踢轮胎

在这一点上,Project Loom仍处于探索阶段。 API会不断变化,因此在假期过后尝试使用该代码时,请准备好适应最新的API版本。

您可以从http://jdk.java.net/loom/下载Project Loom的二进制文件,但是它们很少更新。 但是,在Linux机器或VM上,自己构建最新版本很容易:

git clone https://github.com/openjdk/loom
cd loom 
git checkout fibers
sh configure  
make images

根据已安装的内容, configure可能会出现一些故障,但是消息会告诉您需要安装哪些软件包以便继续进行。

在API的当前版本中,光纤或现在称为虚拟线程的虚拟线程表示为Thread类的对象。 这是三种生产纤维的方法。 首先,有一个新的工厂方法可以构造OS线程或虚拟线程:

Thread thread = Thread.newThread(taskname, Thread.VIRTUAL, runnable);

如果您需要更多自定义,则有一个构建器API:

Thread thread = Thread.builder().name(taskname).virtual().priority(Thread.MAX_PRIORITY).task(runnable).build();

但是,一段时间以来,手动创建线程一直被认为是较差的做法,因此您可能不应执行任何一种操作。 而是将执行程序与线程工厂一起使用:

ThreadFactory factory = Thread.builder().virtual().factory();
ExecutorService exec = Executors.newFixedThreadPool(NTASKS, factory);

现在,熟悉的固定线程池将以与以往相同的方式从工厂调度虚拟线程。 当然,还将有OS级别的载体线程来运行这些虚拟线程,但这是虚拟线程实现的内部。

固定线程池将限制并发虚拟线程的总数。 默认情况下,从虚拟线程到承载线程的映射是通过使用系统属性jdk.defaultScheduler.parallelism或默认情况下Runtime.getRuntime().availableProcessors()所给定数量的内核的jdk.defaultScheduler.parallelism池完成的。 您可以在线程工厂中提供自己的调度程序:

factory = Thread.builder().virtual().scheduler(myExecutor).factory();

我不知道这是否是人们想要做的。 为什么载具线程多于核心?

返回我们的执行人服务。 您可以在虚拟线程上执行任务,就像在OS级线程上执行任务一样:

for (int i = 1; i <= NTASKS; i++) {String taskname = "task-" + i;exec.submit(() -> run(taskname));
}
exec.shutdown();
exec.awaitTermination(delay, TimeUnit.MILLISECONDS);

作为一个简单的测试,我们可以在每个任务中入睡。

public static int DELAY = 10_000;public static void run(Object obj) {try {Thread.sleep((int) (DELAY * Math.random()));} catch (InterruptedException ex) {ex.printStackTrace();}System.out.println(obj);}

如果现在将NTASKS设置为1_000_000并在工厂生成器中.virtual() ,则该程序将失败,并显示内存不足错误。 一百万个OS级线程占用大量内存。 但是使用虚拟线程,它可以工作。

至少,它应该工作,并且对我之前的Loom版本确实有效。 不幸的是,在12月5日下载的构建中,我得到了一个核心转储。 当我尝试使用Loom时,这时有发生。 希望在您尝试此操作时可以解决此问题。

现在,您可以尝试更复杂的事情了。 亨氏·卡布兹(Heinz Kabutz)最近为益智游戏提供了一个程序,该程序可加载数千个Dilbert卡通图像。 对于每个日历日,都有一个页面,例如https://dilbert.com/strip/2011-06-05 。 程序读取这些页面,在每个页面中找到卡通图像的URL,然后加载每个图像。 这是一堆混乱的期货 ,有点像:

CompletableFuture.completedFuture(getUrlForDate(date)).thenComposeAsync(this::readPage, executor).thenApply(this::getImageUrl).thenComposeAsync(this::readPage).thenAccept(this::process);

使用光纤,代码更加清晰:

exec.submit(() -> {      String page = new String(readPage(getUrlForDate(date)));byte[] image = readPage(getImageUrl(page));process(image);
});

当然,每个对readPage的调用readPage块,但是对于纤维,我们不在乎。

尝试一下您关心的事情。 阅读大量的网页,进行处理,进行更多的阻塞读取,并享受光纤阻塞便宜的事实。

结构化的一致性

Project Loom的最初动机是实现光纤,但是今年早些时候,该项目开始了针对结构化并发的实验性API。 在这篇强烈推荐的文章 (从中拍摄以下图像)中,Nathaniel Smith提出了结构化的并发形式。 这是他的中心论点。 在新线程中启动任务实际上并不比使用GOTO编程好,即有害:

new Thread(runnable).start();

当多个线程在没有协调的情况下运行时,这将是意大利面条代码。 在1960年代,结构化编程将goto替换为分支,循环和函数:

现在,结构化并发的时机已经到来。 当启动并发任务时,通过阅读程序文本,我们应该知道它们何时完成。

这样,我们可以控制任务使用的资源。

到2019年夏季,Project Loom有了一个用于表达结构化并发的API。 不幸的是,由于最近进行了统一线程和光纤API的实验,该API目前处于混乱状态,但是您可以通过http://jdk.java.net/loom/上的原型进行尝试。

在这里,我们安排了许多任务:

FiberScope scope = FiberScope.open();
for (int i = 0; i < NTASKS; i++) {scope.schedule(() -> run(i));
}
scope.close();

调用scope.close()阻塞,直到所有光纤完成。 请记住,光纤阻塞不是问题。 一旦关闭示波器,您就可以确定光纤已经完成。

FiberScope是可FiberScope的,因此您可以使用try -with-resources语句:

try (var scope = FiberScope.open()) {...
}

但是,如果其中一项任务没有完成怎么办?

您可以使用截止日期( Instant )或超时( Duration )创建范围:

try (var scope = FiberScope.open(Instant.now().plusSeconds(30))) {for (...)scope.schedule(...);
}

截止期限/超时之前尚未完成的所有光纤都将被取消。 怎么样? 继续阅读。

消除

取消一直是Java的痛苦。 按照惯例,您可以通过中断线程来取消线程。 如果线程正在阻塞,则阻塞操作以InterruptedException终止。 否则,设置中断状态标志。 正确地进行检查是乏味的。 可以重置中断状态,或者InterruptedException是已检查的异常,这没有帮助。

java.util.concurrent中取消的处理一直不一致。 考虑ExecutorService.invokeAny 。 如果任何任务产生结果,则其他任务将被取消。 但是CompletableFuture.anyOf允许所有任务运行完成,即使其结果将被忽略。

2019年夏季的Project Loom API解决了取消问题。 在该版本中,光纤具有cancel操作,类似于interrupt ,但是取消是不可撤销的。 如果当前光纤已被取消,则静态Fiber.cancelled方法将返回true

当示波器超时时,其光纤将被取消。

取消可以由FiberScope构造函数中的以下选项控制。

  • CANCEL_AT_CLOSE :关闭范围取消所有调度的光纤而不是阻塞
  • PROPAGATE_CANCEL :如果取消拥有光纤,则任何新调度的光纤都会自动取消
  • IGNORE_CANCEL :无法取消预定的光纤

所有这些选项都未在顶层设置。 PROPAGATE_CANCELIGNORE_CANCEL选项是从父范围继承的。

如您所见,有相当多的可调整性。 我们将不得不看到重新审视此问题时会发生什么。 对于结构化并发,当示波器超时或被强制关闭时,必须自动取消示波器中的所有光纤。

螺纹局部

让我感到惊讶的是,Project Loom实现者的痛苦之一是ThreadLocal变量,以及更深奥的东西-上下文类加载器AccessControlContext 。 我不知道有那么多东西骑在线程上。

如果您的数据结构不适合并发访问,则有时可以在每个线程中使用一个实例。 经典示例是SimpleDateFormat 。 当然,您可以继续构造新的格式化程序对象,但这并不高效。 所以你想分享一个。 但是全球

public static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");

将无法正常工作。 如果两个线程同时访问它,则格式可能会混乱。

因此,每个线程中有一个是有意义的:

public static final ThreadLocal<SimpleDateFormat> dateFormat= ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd"));

要访问实际的格式化程序,请致电

String dateStamp = dateFormat.get().format(new Date());

首次调用get时,将调用构造函数中的lambda。 从那时起,get方法返回属于当前线程的实例。

对于线程,这是公认的做法。 但是,当一百万个光纤存在时,您真的想拥有一百万个实例吗?

这对我来说不是问题,因为使用线程安全的东西(如java.time格式化程序)似乎更容易。 但是Project Loom一直在考虑“范围本地”对象-那些FiberScope被重新激活了。

在线程与处理器数量一样多的情况下,线程局部变量也已被用作处理器局部性的近似值。 可以实际模拟用户意图的API可以支持此功能。

项目状况

想要使用Project Loom的开发人员自然会沉迷于API,如您所见,该API尚未解决。 但是,很多实施工作都处于幕后。

一个关键部分是在操作阻塞时使光纤停放。 已经完成了网络连接,因此您可以在光纤内连接到网站,数据库等。 当前不支持本地文件操作块时的停车。

实际上,在JDK 11、12和13中已经重新实现了这些库,这是对频繁发布实用程序的致敬。

目前尚不支持在监视器上进行阻塞( synchronized块和方法),但最终需要这样做。 ReentrantLock现在可以了。

如果纤维以本机方法阻塞,则将“固定”线程,并且所有纤维都不会前进。 Project Loom对此无能为力。

Method.invoke需要更多工作才能得到支持。

有关调试和监视支持的工作正在进行中。

如前所述,稳定性仍然是一个问题。

最重要的是,性能还有一段路要走。 停放光纤不是免费的午餐。 每次都需要替换运行时堆栈的一部分。

在所有这些方面都取得了很大的进展,所以让我们回顾一下开发人员关心的API。 现在是查看Project Loom并考虑如何使用它的好时机。

同一类代表线和纤维对您有价值吗? 还是您希望将一些Thread行李搬走? 您是否认同结构化并发的承诺?

试一下Project Loom,看看它如何与您的应用程序和框架一起工作,并为无畏的开发团队提供反馈!

翻译自: https://www.javacodegeeks.com/2019/12/project-loom.html

织机原理

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

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

相关文章

java集合——java.util.Properties类

【0】README 0.1&#xff09;以下全文转自 &#xff1a;  http://trans.blog.51cto.com/503170/110227/ 【1】认识properties文件 1、 properties文件是一个文本文件2、 properties文件的语法有两种&#xff0c;一种是注释&#xff0c;一种属性配置。 注 释&#xff1a;前…

电力系统潮流计算matlab程序,大神们,求个电力系统潮流计算的matlab程序。

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼ninput(Please input n\n); %n表示系统的节点数d0input(Please input d0\n); %d0表示系统的误差minput(Please inpit m\n); %m表示系统的pq节点的个数p0ones(n-1,1);q0ones(m,1);d1;dpvones(n-1,1);dqvones(m,1);dpzeros(n-1,1);dq…

java之正则表达式

一、匹配字符串String类的matchs方法用的是正则表达式匹配&#xff0c;matchs方法与equals类似。"javac".matchs("javac");"javac".equals("javac");结果都为true。match方法更强大&#xff0c;不仅能匹配固定字符串&#xff0c;还能匹…

gradle配置_Gradle配置

gradle配置在这篇文章中&#xff0c;我们将介绍有关Gradle配置的综合文章。 1.技术 Gradle是用于自动任务管理的开源构建工具。 它基于Apache Ant&#xff0c;Apache Maven的概念。 Gradle是使用Groovy语言开发的。 Gradle是使用基于Groovy的领域特定语言&#xff08;DSL&…

removeAll throws java.lang.UnsupportedOperationException

【1】出现的问题&#xff1a; 如果该列表list 不支持 removeAll 方法的话&#xff0c; 就会抛出UnsupportedOperationException 【2】解决方法&#xff1a; List<String> list new ArrayList(Arrays.asList(new String[]{"C","B","D"})…

C#的float、double与decimal

float 单精度浮点 32bit&#xff0c;double 双精度浮点64bit&#xff0c;decimal是高精度 128bit&#xff0c;浮点型。 float double 是 基本类型&#xff08;primitive type&#xff09;&#xff0c;decimal不是。 float 有效数字7位&#xff0c;范围 1.5 10E−45 to 3.4 …

matlab支持 编程语言,用于数学的10个优秀编程语言

原标题&#xff1a;用于数学的10个优秀编程语言译文&#xff1a;http://www.codeceo.com/article/10-programming-language-for-math.html英文&#xff1a;10 Great Programming Languages for Mathematics翻译&#xff1a;码农网 – 小峰作为一个对数学和编程语言充满激情的人…

java锁_Java锁

java锁锁是一种线程同步机制&#xff0c;例如同步块。 锁是使用同步块在内部实现的。 因此&#xff0c;我们可以在Java中使用锁代替同步关键字。 锁比同步块更灵活&#xff0c;更复杂。 从Java 5版本开始&#xff0c;JDK提供了几种锁实现&#xff0c;例如ReentrantReadWriteLo…

java线程——阻塞队列

【0】README 0.1&#xff09;本文均转自两篇网络文章&#xff0c; 转载地址在转载处做了标记&#xff0c; 旨在理清 “阻塞队列”的相关知识 &#xff1b; 【1】转自&#xff1a; http://www.cnblogs.com/dolphin0520/p/3932906.html 1.1&#xff09;在前面我们接触的队列都是…

C#接口

接口是指定一组函数成员而不实现成员的引用类型&#xff0c;其他类型-类和接口可以实现接口。1. 接口是一个引用类型&#xff0c;通过接口可以实现多重继承。2. C#中接口的成员不能有new、public、protected、internal、private等修饰符。3. 接口中只能声明"抽象"成员…

消费者驱动的契约测试_告诉我们您想要什么,我们将做到:消费者驱动的合同测试消息传递...

消费者驱动的契约测试相当早以前&#xff0c;我们从REST&#xff08;ful&#xff09; Web API的角度讨论了消费者驱动的合同测试 &#xff0c;尤其是将其投射到Java&#xff08; JAX-RS 2.0规范&#xff09;的角度。 可以公平地说&#xff0c;至少在公共API方面&#xff0c; RE…

linux将屏幕输出到文件,Linux命令执行的屏幕输出内容重定向到日志文件

摘要: 作者&#xff1a;Syn良子 出处&#xff1a;http://www.cnblogs.com/cssdongl 转载请注明出处快速mark一下这个命令细节&#xff0c;免得以后使用又忘记了大家都知道可以用echo来输出内容到文件&#xff0c;比如 echo “hello,word”>> temp.txt那么我现在想把hadoo…

java线程——信号量(Semaphore)+障栅(CyclicBarrier)

【0】README 0.1&#xff09;以下内容转自网络&#xff0c;旨在理清 “java线程——信号量&#xff08;Semaphore&#xff09;障栅&#xff08;CyclicBarrier&#xff09;”的相关知识 【1】信号量 1.1&#xff09;信号量定义&#xff1a;从概念上讲&#xff0c;一个信号量管…

C#的foreach

一、foreach语句 C#的foreach 语句不会解析为IL 代码中的foreach 语句。 C#编译器会把foreach 语句转换为 IEnumerable 接口的方法和属性。 下面是一个简单的foreach 语句&#xff0c;它迭代数组中的所有元素&#xff0c;并逐个显示它们&#xff1a; foreach (A a in aas)…

linux中echo命令不输出换行,shell脚本echo输出不换行功能增强实例

这是第8/101个脚本There are as many ways to solve this quirky echo problem as there are pages in this book. One of my favorites is very succinct:function echon{echo "$*" | awk { printf "%s" $0 }}You may prefer to avoid the overhead incur…

java联接pg库_成为Java流大师–第5部分:将联接的数据库表转换为流

java联接pg库是否可以将联接的数据库表转换为Java Stream&#xff1f; 答案是肯定的。 由于我们已经多次提出这个问题&#xff0c;因此我们决定写另一篇动手实验文章&#xff0c;说明如何执行更高级的Stream Joins。 因此&#xff0c;这里是第六篇中的第五篇&#xff0c;后面还…

java线程——什么是线程?

【0】README 0.1&#xff09; 本文描述转自 core java volume 1&#xff0c; 源代码为原创&#xff0c;旨在理解 java线程——什么是线程&#xff1f; 的相关知识&#xff1b; 0.2&#xff09;线程定义&#xff1a;一个程序可以执行多个任务&#xff0c;每一个任务成为线程&a…

C#的类修饰符和成员修饰符

一、类修饰符 类修饰符&#xff1a;public、internal、 partial、abstract、sealed、static、new、protected、private、protected internal1、public&#xff1a;访问不受限制的&#xff0c;所有的本程序集以及其他的程序集里面的类都能够访问2、internal&#xff1a;本程序集…

linux 端口方法防火墙,Linux 打开端口方法(防火墙操作)

Linux防火墙操作(经测试部分命令无效)关闭防火墙:service iptables stop开启防火墙:service iptables start防火墙状态:service iptables status永久关闭:chkconfig iptables off永久开启:chkconfig iptables on方法一(命令):1. 开放端口命令&#xff1a; /sbin/iptables -I IN…

kafka 发布-订阅模式_使用Apache Kafka作为消息系统的发布-订阅通信中的微服务,并通过集成测试进行了验证...

kafka 发布-订阅模式发布-订阅消息系统在任何企业体系结构中都起着重要作用&#xff0c;因为它可以实现可靠的集成而无需紧密耦合应用程序。 在解耦的系统之间共享数据的能力并不是一个容易解决的问题。 考虑一个企业&#xff0c;其中具有使用不同语言和平台独立构建的多个应用…