java11又出新版本了,我还在java8上停着。不过这也挡不住我对他的热爱,忍不住查看了一下他的新性能,由于自己知识有限,只总结了以下八个特性;
1、本地变量类型推断
什么是局部变量类型推断?
var javastack = "javastack";
system.out.println(javastack);
大家看出来了,局部变量类型推断就是左边的类型直接使用 var 定义,而不用写具体的类型,编译器能根据右边的表达式自动推断类型,如上面的 string 。
var javastack = "javastack";
就等于:
string javastack = "javastack";
2、字符串加强
java 11 增加了一系列的字符串处理方法,如以下所示。
// 判断字符串是否为空白
" ".isblank(); // true
// 去除首尾空格
" javastack ".strip(); // "javastack"
// 去除尾部空格
" javastack ".striptrailing(); // " javastack"
// 去除首部空格
" javastack ".stripleading(); // "javastack "
// 复制字符串
"java".repeat(3);// "javajavajava"
// 行数统计
"a\nb\nc".lines().count(); // 3
3、集合加强
自 java 9 开始,jdk 里面为集合(list/ set/ map)都添加了 of 和 copyof 方法,它们两个都用来创建不可变的集合,来看下它们的使用和区别。
示例1:
var list = list.of("java", "python", "c");
var copy = list.copyof(list);
system.out.println(list == copy); // true
示例2:
var list = new arraylist();
var copy = list.copyof(list);
system.out.println(list == copy); // false
示例1和2代码差不多,为什么一个为true,一个为false?
来看下它们的源码:
static list of(e... elements) {
switch (elements.length) { // implicit null check of elements
case 0:
return immutablecollections.emptylist();
case 1:
return new immutablecollections.list12<>(elements[0]);
case 2:
return new immutablecollections.list12<>(elements[0], elements[1]);
default:
return new immutablecollections.listn<>(elements);
}
}
static list copyof(collection coll) {
return immutablecollections.listcopy(coll);
}
static list listcopy(collection coll) {
if (coll instanceof abstractimmutablelist && coll.getclass() != sublist.class) {
return (list)coll;
} else {
return (list)list.of(coll.toarray());
}
}
可以看出 copyof 方法会先判断来源集合是不是 abstractimmutablelist 类型的,如果是,就直接返回,如果不是,则调用 of 创建一个新的集合。
示例2因为用的 new 创建的集合,不属于不可变 abstractimmutablelist 类的子类,所以 copyof 方法又创建了一个新的实例,所以为false.
注意:使用of和copyof创建的集合为不可变集合,不能进行添加、删除、替换、排序等操作,不然会报 java.lang.unsupportedoperationexception 异常。
上面演示了 list 的 of 和 copyof 方法,set 和 map 接口都有。
4、stream 加强
stream 是 java 8 中的新特性,java 9 开始对 stream 增加了以下 4 个新方法。
1) 增加单个参数构造方法,可为null
stream.ofnullable(null).count(); // 0
2) 增加 takewhile 和 dropwhile 方法
stream.of(1, 2, 3, 2, 1)
.takewhile(n -> n < 3)
.collect(collectors.tolist()); // [1, 2]
从开始计算,当 n < 3 时就截止。
stream.of(1, 2, 3, 2, 1)
.dropwhile(n -> n < 3)
.collect(collectors.tolist()); // [3, 2, 1]
这个和上面的相反,一旦 n < 3 不成立就开始计算。
3)iterate重载
这个 iterate 方法的新重载方法,可以让你提供一个 predicate (判断条件)来指定什么时候结束迭代。
如果你对 jdk 8 中的 stream 还不熟悉,可以看之前分享的这一系列教程。
5、optional 加强
opthonal 也增加了几个非常酷的方法,现在可以很方便的将一个 optional 转换成一个 stream, 或者当一个空 optional 时给它一个替代的。
optional.of("javastack").orelsethrow(); // javastack
optional.of("javastack").stream().count(); // 1
optional.ofnullable(null)
.or(() -> optional.of("javastack"))
.get(); // javastack
6、inputstream 加强
inputstream 终于有了一个非常有用的方法:transferto,可以用来将数据直接传输到 outputstream,这是在处理原始数据流时非常常见的一种用法,如下示例。
var classloader = classloader.getsystemclassloader();
var inputstream = classloader.getresourceasstream("javastack.txt");
var javastack = file.createtempfile("javastack2", "txt");
try (var outputstream = new fileoutputstream(javastack)) {
inputstream.transferto(outputstream);
}
7、http client api
这是 java 9 开始引入的一个处理 http 请求的的孵化 http client api,该 api 支持同步和异步,而在 java 11 中已经为正式可用状态,你可以在 java.net 包中找到这个 api。
来看一下 http client 的用法:
var request = httprequest.newbuilder()
.uri(uri.create("https://javastack.cn"))
.get()
.build();
var client = httpclient.newhttpclient();
// 同步
httpresponse response = client.send(request, httpresponse.bodyhandlers.ofstring());
system.out.println(response.body());
// 异步
client.sendasync(request, httpresponse.bodyhandlers.ofstring())
.thenapply(httpresponse::body)
.thenaccept(system.out::println);
上面的 .get() 可以省略,默认请求方式为 get!
更多使用示例可以看这个 api,后续有机会再做演示。
现在 java 自带了这个 http client api,我们以后还有必要用 apache 的 httpclient 工具包吗?
8、化繁为简,一个命令编译运行源代码
看下面的代码。
// 编译
javac javastack.java
// 运行
java javastack
在我们的认知里面,要运行一个 java 源代码必须先编译,再运行,两步执行动作。而在未来的 java 11 版本中,通过一个 java 命令就直接搞定了,如以下所示。
java javastack.java