Java SE 12 新增特性

作者:Grey

原文地址:Java SE 12 新增特性

源码

源仓库: Github:java_new_features

镜像仓库: GitCode:java_new_features

String 类中新增的 API

Java SE 12内置了这样一个方法:String.indent()。下面的例子显示了如何将一个多行字符串缩进四个空格。

package git.snippets.jdk12;

/**
* String 新增API
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2022/8/19
* @since 12
*/
public class StringNewAPIDemo {
public static void main(String[] args) {
String s = "I am\na multiline\nString.";
System.out.println(s);
System.out.println(s.indent(4));
}
}

输出结果

I am
a multiline
String.
I am
a multiline
String.Code language: MIPS Assembly (mipsasm)

String.transform()方法将一个任意的函数应用于一个字符串

package git.snippets.jdk12;

import java.math.BigDecimal;

/**
* String 新增API
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2022/8/19
* @since 12
*/
public class StringNewAPIDemo {
public static void main(String[] args) { // 将一个任意的函数应用于一个字符串
// 效果等同于 String uppercase = "abcde".toUpperCase();
String uppercase = "abcde".transform(String::toUpperCase);
System.out.println(uppercase); // 效果等同于 Integer i = Integer.valueOf("12345");
Integer i = "12345".transform(Integer::valueOf);
System.out.println(i); // 效果等同于 BigDecimal big = new BigDecimal("1234567891011121314151617181920");
BigDecimal big = "1234567891011121314151617181920".transform(BigDecimal::new);
System.out.println(big);
}
}

输出:

ABCDE
12345
1234567891011121314151617181920

File 增强 API

你可以使用Files.mismatch()方法来比较两个文件的内容。

如果两个文件是一样的,该方法返回 -1 。否则,它返回两个文件不同的第一个字节的位置。如果其中一个文件在检测到差异之前结束,则返回该文件的长度。示例代码如下

package git.snippets.jdk12;

import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption; /**
* 文件内容对比
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2021/11/29
* @since 12
*/
public class FileMisMatchTest {
public static void main(String[] args) throws IOException {
match();
} static void match() throws IOException {
Path pathA = Files.createFile(Paths.get("a.txt"));
Path pathB = Files.createFile(Paths.get("b.txt")); // 写入相同内容
Files.write(pathA, "abc".getBytes(), StandardOpenOption.WRITE);
Files.write(pathB, "abc".getBytes(), StandardOpenOption.WRITE);
long mismatch = Files.mismatch(pathA, pathB);
System.out.println(mismatch); // 追加不同内容
Files.write(pathA, "1234".getBytes(), StandardOpenOption.APPEND);
Files.write(pathB, "1321".getBytes(), StandardOpenOption.APPEND);
mismatch = Files.mismatch(pathA, pathB);
System.out.println(mismatch); // 删除创建的文件
pathA.toFile().deleteOnExit();
pathB.toFile().deleteOnExit();
}
}

分流收集器

对于某些要求,你可能想用两个收集器来终止一个流,而不是一个,并将两个收集器的结果结合起来。

比如,获取随机数流中的最大值和最小值

运行后,报错

Exception in thread "main" java.lang.IllegalStateException: stream has already been operated upon or closed
at java.base/java.util.stream.AbstractPipeline.evaluate(AbstractPipeline.java:229)
at java.base/java.util.stream.ReferencePipeline.collect(ReferencePipeline.java:578)
at git.snippets.jdk12.TeeCollectorTest.main(TeeCollectorTest.java:19)

通过如上日志可以看到:我们只能终止一个流一次。

那么我们如何解决这个问题呢?

一种方法是写一个自定义的收集器,将最小值和最大值累积到一个2元素的int数组中。

package git.snippets.jdk12;

import java.util.Random;
import java.util.stream.Collectors;
import java.util.stream.Stream; /**
* 分流收集器
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2022/8/19
* @since 12
*/
public class TeeCollectorTest {
public static void main(String[] args) {
// 写一个自定义的收集器,将最小值和最大值累积到一个2元素的int数组中。
Stream<Integer> numbers = new Random().ints(100).boxed(); int[] result = numbers.collect(() -> new int[]{Integer.MAX_VALUE, Integer.MIN_VALUE}, (minMax, i) -> {
if (i < minMax[0]) {
minMax[0] = i;
}
if (i > minMax[1]) {
minMax[1] = i;
}
}, (minMax1, minMax2) -> {
if (minMax2[0] < minMax1[0]) {
minMax1[0] = minMax2[0];
}
if (minMax2[1] > minMax1[1]) {
minMax1[1] = minMax2[1];
}
}); long range = (long) result[1] - result[0];
System.out.println(range);
}
}

这种方法相当复杂,我们可以使用Java SE 12中引入的 "Teeing Collector "来做。我们可以指定两个收集器(称为下游收集器)和一个合并函数,将两个收集器的结果合并。代码如下

package git.snippets.jdk12;

import java.util.Random;
import java.util.stream.Collectors;
import java.util.stream.Stream; /**
* 分流收集器
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2022/8/19
* @since 12
*/
public class TeeCollectorTest {
public static void main(String[] args) { // 方法2,使用Tee Collector
Stream<Integer> numbers = new Random().ints(100).boxed(); long range = numbers.collect(Collectors.teeing(Collectors.minBy(Integer::compareTo), Collectors.maxBy(Integer::compareTo), (min, max) -> (long) max.orElseThrow() - min.orElseThrow()));
System.out.println(range);
}
}

新版 Switch 使用方式

我们现在可以通过用逗号分隔多个case来简化switch语句,并使用箭头符号来消除容易出错的中断语句,这个特性在Java SE 12中是预览特性,在Java SE 14正式引入。代码如下:

注:如果你用Java SE 12运行上述代码,需要指定--enable-preview参数,如果使用 Intellij IDEA ,参考How to Enable Java Preview Features and Run Code from IntelliJ IDEA

package git.snippets.jdk12;

package git.snippets.jdk12;

/**
* switch新用法
* 预览特性,在jdk14正式引入
* IDEA 启用 预览功能
* <p>
* <a href="https://foojay.io/today/how-to-run-project-loom-from-intellij-idea/">how to run project loom from intellij idea</a>
* <p>
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2021/11/29
* @since 12
*/
public class NewSwitch {
public static void main(String[] args) {
test("apple");
test2("march");
} // 在jdk12是预览特性,在14正式引入
static void test(String c) {
switch (c) {
case "apple", "Apple" -> System.out.println("苹果");
case "banana", "Banana" -> System.out.println("香蕉"); }
} static void test2(String day) {
String season = switch (day) {
case "march", "april", "may" -> "春天";
case "june", "july", "august" -> "夏天";
case "september", "october", "november" -> "秋天";
case "december", "january", "february" -> "冬天";
default -> {
throw new RuntimeException("day error");
}
};
System.out.println("当前季节是:" + season);
} }

简化数字格式显示

使用静态方法NumberFormat.getCompactNumberInstance(),我们可以创建一个格式化器。这是一种便于人类阅读的格式,如 "2M "或 "30亿"。

代码见:

package git.snippets.jdk12;

import java.text.NumberFormat;
import java.util.Locale; /**
* 简化的数字格式可以直接转换数字显示格式,比如 1000 -> 1K,1000000 -> 1M 。
*
* @author <a href="mailto:410486047@qq.com">Grey</a>
* @date 2021/11/29
* @since 12
*/
public class CompactNum {
public static void main(String[] args) {
test();
test2();
test3();
} static void test() {
System.out.println("Compact Formatting is:");
NumberFormat upvotes = NumberFormat.getCompactNumberInstance(Locale.US, NumberFormat.Style.SHORT); System.out.println(upvotes.format(100));
System.out.println(upvotes.format(1000));
System.out.println(upvotes.format(10000));
System.out.println(upvotes.format(100000));
System.out.println(upvotes.format(1000000)); // 设置小数位数
upvotes.setMaximumFractionDigits(1);
System.out.println(upvotes.format(1234));
System.out.println(upvotes.format(123456));
System.out.println(upvotes.format(12345678));
System.out.println(upvotes.format(123456789));
} static void test2() {
System.out.println("Compact Formatting is:");
NumberFormat upvotes = NumberFormat.getCompactNumberInstance(Locale.CHINA, NumberFormat.Style.SHORT); System.out.println(upvotes.format(100));
System.out.println(upvotes.format(1000));
System.out.println(upvotes.format(10000));
System.out.println(upvotes.format(100000));
System.out.println(upvotes.format(1000000)); // 设置小数位数
upvotes.setMaximumFractionDigits(1);
System.out.println(upvotes.format(1234));
System.out.println(upvotes.format(123456));
System.out.println(upvotes.format(12345678));
System.out.println(upvotes.format(123456789));
} static void test3() {
System.out.println("Compact Formatting is:");
NumberFormat nfShort =
NumberFormat.getCompactNumberInstance(Locale.US, NumberFormat.Style.SHORT);
NumberFormat nfLong =
NumberFormat.getCompactNumberInstance(Locale.US, NumberFormat.Style.LONG); System.out.println(" 1,000 short -> " + nfShort.format(1_000));
System.out.println(" 456,789 short -> " + nfShort.format(456_789));
System.out.println(" 2,000,000 short -> " + nfShort.format(2_000_000));
System.out.println("3,456,789,000 short -> " + nfShort.format(3_456_789_000L));
System.out.println();
System.out.println(" 1,000 long -> " + nfLong.format(1_000));
System.out.println(" 456,789 long -> " + nfLong.format(456_789));
System.out.println(" 2,000,000 long -> " + nfLong.format(2_000_000));
System.out.println("3,456,789,000 long -> " + nfLong.format(3_456_789_000L));
}
}

输出结果为:

Compact Formatting is:
100
1K
10K
100K
1M
1.2K
123.5K
12.3M
123.5M
Compact Formatting is:
100
1,000
1万
10万
100万
1,234
12.3万
1234.6万
1.2亿
Compact Formatting is:
1,000 short -> 1K
456,789 short -> 457K
2,000,000 short -> 2M
3,456,789,000 short -> 3B 1,000 long -> 1 thousand
456,789 long -> 457 thousand
2,000,000 long -> 2 million
3,456,789,000 long -> 3 billion

更多

Java SE 7及以后各版本新增特性

参考资料

Java 12 Features (with Examples)

official Java 12 Release Notes

Java SE 12 新增特性的更多相关文章

  1. Java SE 8 新增特性

    Java SE 8 新增特性 作者:Grey 原文地址: Java SE 8 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new_ ...

  2. Java SE 13 新增特性

    Java SE 13 新增特性 作者:Grey 原文地址:Java SE 13 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

  3. Java SE 14 新增特性

    Java SE 14 新增特性 作者:Grey 原文地址:Java SE 14 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

  4. Java SE 15 新增特性

    Java SE 15 新增特性 作者:Grey 原文地址:Java SE 15 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

  5. Java SE 9 新增特性

    Java SE 9 新增特性 作者:Grey 原文地址: Java SE 9 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new_ ...

  6. Java SE 10 新增特性

    Java SE 10 新增特性 作者:Grey 原文地址:Java SE 10 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

  7. Java SE 11 新增特性

    Java SE 11 新增特性 作者:Grey 原文地址:Java SE 11 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

  8. Java SE 16 新增特性

    Java SE 16 新增特性 作者:Grey 原文地址:Java SE 16 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

  9. Java SE 17 新增特性

    Java SE 17 新增特性 作者:Grey 原文地址:Java SE 17 新增特性 源码 源仓库: Github:java_new_features 镜像仓库: GitCode:java_new ...

随机推荐

  1. 从数字化概念到落地,都说是一道坎,JNPF能为企业带来什么呢?​

    数字经济席卷全球推动产业转型是必然趋势,人类社会正在数字化大潮中发生深刻变革,数字化越来越成为推动经济社会发展的核心驱动力.企业正处于数字经济大潮的风口浪尖,故企业经验决策者应深刻认识数字化转型对于企 ...

  2. Aged-cat 的 WebServer 配置详细流程

    前言 最近看到一个不错的webserver项目,但是配置过程给的不详细,这里给出详细配置过程 项目地址:https://github.com/Aged-cat/WebServer (感谢老猫大神的项目 ...

  3. 用STM32玩L298N(正反转、调速)

    目录 用STM32玩L298N(正反转.调速) 控制直流电机正反转 使用PWM调速 用STM32玩L298N(正反转.调速) 开发板:STM32F103ZET6(正点原子F103核心板)/STM32F ...

  4. 支付宝开放平台--网页&移动应用(一)

    前提是先在支付宝上签约自己需要的支付宝功能,然后支付宝开放平台才能设置你需要的功能 一:支付宝开放平台登录 登录进入支付宝开放平台 二:根据自己的需求创建应用(我是用的网页&移动应用) 三:点 ...

  5. C# · 委托语句简化演变

    1.委托基础语句形式 namespace QLVision { delegate void dHelp();//定义委托 static class Program { /// <summary& ...

  6. 【黑马pink老师的H5/CSS课程】(一)基本介绍

    视频P4~P6 黑马程序员pink老师前端入门教程,零基础必看的h5(html5)+css3+移动 1.网页 1.1 什么是网页 网站是网页的集合,网页是构成网站的基本元素,常用html格式文件 1. ...

  7. Qt数据可视化(散点图、折线图、柱状图、盒须图、饼状图、雷达图)开发实例

    ​  目录 散点图 折线图 柱状图 水平柱状图 水平堆叠图 水平百分比柱状图 盒须图 饼状图 雷达图 Qt散点图.折线图.柱状图.盒须图.饼状图.雷达图开发实例. 在开发过程中我们会使用多各种各样的图 ...

  8. 全新升级的AOP框架Dora.Interception[6]: 框架设计和实现原理

    本系列前面的五篇文章主要介绍Dora.Interception(github地址,觉得不错不妨给一颗星)的编程模式以及对它的扩展定制,现在我们来聊聊它的设计和实现原理.(拙著<ASP.NET C ...

  9. 如何用python做出老师看了都给满分的GUI学生管理系统毕设

    序 言 哈喽大家好鸭!我是小熊猫 最近有什么有趣的事情发生吗?快来说给我听听( •̀ ω •́ )✧表弟大学快毕业了,学了一个学期Python居然还不会写学生管理系统,真的给我丢脸啊,教他又不肯学,还 ...

  10. go-zero微服务实战系列(八、如何处理每秒上万次的下单请求)

    在前几篇的文章中,我们花了很大的篇幅介绍如何利用缓存优化系统的读性能,究其原因在于我们的产品大多是一个读多写少的场景,尤其是在产品的初期,可能多数的用户只是过来查看商品,真正下单的用户非常少.但随着业 ...