Java8之前版本的特性比较

转载:王爵nice

作者: Ian | 2018-07-10 | 阅读

转载:王爵nice – https://zhuanlan.zhihu.com/p/28160344

引言

现在JDK的版本已经很高了,但是企业主流应该大多数都是JDK8及以下。虽然我们开始了Java8的旅程,但是很多人直接从Java6上手了Java8,Z也是的,JDK7 好像没用过😅 也许有些JDK7的特性你还不知道,在本篇中一起回顾一下我们忘记了的那些特性。尽管不可能所有的特性都回顾一遍,但是常用的核心的可以一起学习。

异常改进

try-with-resources 这个特性是在JDK7中出现的,我们在之前操作一个流对象的时候大概是这样的:

try{
    //使用流对象
    stream.ream();
    stream.write();
}catch(Exception e){
    //捕获异常 并处理
}finally{
    //关闭流资源 写在这个是里面的代码是必须执行的
	if(null != stream){
        stream.close();
	}
}

这样无疑有些繁琐,而且finally块还有可能抛出异常。在JDK7中提出了try-with-resources机制,它规定你操作的类只要是实现了AutoCloseable接口就可以在try语句块退出的时候自动调用close方法关闭流资源。

public static void tryWithResources()throws IOException(){
    try(InputStream ins = new FileInputStream("/home/Ian/a.txt")){
        char charStr = (char)ins.read();
        System.out.print(charStr);
    }
}

使用多个资源

try(InputStream is = new FileInputStream("/home/ian/a.txt");
	OutputStream os = new FileOutputStream("/home/ian/b.txt")
){
	char charStr = (char)is.read();
	os.write(charStr);
}

当如如果你使用的是非标准库的类也可以自定义AutoCloseable,只要实现其close方法即可。

捕获多个Exception

当我们在操作一个对象的时候,有时候它会抛出多个异常,像这样:

try{
    Thread.sleep(20000);
    FileInputStream fis = new FileInputStream("/a/b.txt");
}catch(InterruptedException e){
    e.printStackTrace();
}catch(IOException ee){
	ee.printStackTrace();
}

这样代码写起来要捕获很多异常,不是很优雅,JDK7允许你捕获多个异常:

try{
    Thread.sleep(20000);
    FileInputStream fis = new FileInputStream("/a/b.txt");
}catch(InterruptedException | IOExcepiton e){
    e.printStackTrace();
}

并且catch语句后面的异常参数是final的,不可以再修改、复制。

处理反射异常

使用过反射的朋友可能知道我们有时候操作反射方法的时候会抛出很多不相关的检查异常,例如:

try{
    Class<?> clazz = Class.forName("com.ian.apple.User");
    clazz.getMethods()[0].invoke(object);
}catch(IllegalAccessException e){
    e.printStackTrace();
}catch(InvocationTargetException ee){
	ee.printStackTrace();
}catch(ClassNotFoundException eee){
    eee.printStackTrace();
}

尽管你可以使用catch多个异常的方法将上述异常都捕获,但这也让人感到痛苦。 JDK7修复了这个缺陷,引入一个新类ReflectiveOperationException可以帮你捕获这些反射异常:

/*   
	forName是一个静态方法,其作用:通过调用来获取类名对应的Class对象,同时将Class对象加载进来。
   
   invoke方法
   作用:调用包装在当前Method对象中的方法。
   原型:Object invoke(Object obj,Object...args)
   参数解释:obj:实例化后的对象
           args:用于方法调用的参数
   返回:根据obj和args调用的方法的返回值
*/
try{
    Clss<?> clazz = Class.forName("com.ian.apple.User");
    clazz.getMethods()[0].invoke(object);
}catch(ReflectiveOperationException e){
    e.printStackTrace();
}

文件操作

我们知道在JDK6甚至之前的时候,我们想要读取一个文本文件也是非常麻烦的一件事,而现在他们都变得简单了,这要归功于NIO2,我们先看看之前的做法:

读取一个文本文件

BufferedReader br = null;
try{
    new BufferedReader(new FileReader("file.txt"));
    StringBuilder sb = new StringBuilder();
    String line = br.readLine();
    while(null != line){
        sb.append(line);
        sb.append(System.lineSeparator());
        line = br.readLine();
    }
    String everything = sb.toString();
}catch(Exception e){
	e.printStackTrace();
}finally{
    try{
        br.close();
    }catch(IOExcepiton ee){
        ee.printStackTrace();
    }
}

大家对这样的一段代码一定不陌生,但这样太繁琐了,我们只想读取一个文本文件,要写这么多代码,还要处理让人头大的一堆异常,。。。

下面我要介绍在JDK7中是如何改善这些问题的。

Path

path 用于来表示文件路径和文件,和File对象类似,path对象并不一定要对应一个实际存在在文件,它只是一个路径的抽象序列。

要创建一个path对象有很多种方法,首先是finalpaths的两个static方法,如何从一个路径字符串来构造path对象:

Path path1 = Paths.get("/home/ian","a.txt");
Path path2 = Paths.get("/home/ian/a.txt");
URI u = URI.create("file:////home/ian/a.txt");
Path pathURI = Paths.get(u);

通过FileSystems构造

Path filePath = FileSystems.getDefault().getPath("/home/ian","a.txt");

PathURIFile之间的转换

File file = new File("/home/ian/a.txt");
Path p1 = file.toPath();
p1.toFile();
file.toURI();

读写文件

你可以使用Files类快速实现文件操作,例如读取文件内容:

byte[] data = Files.readAllBytes(Paths.get("/home/ian/a.txt"));
String content = new String(data,StandardCharsets.UTF-8);

如果希望按照行读取文件,可以调用

List<String> lines = Files.readAllLines(Paths.get("/home/ian/a.txt"));

反之你想将字符串写入到文件可以调用

Files.write(Paths.get("/home/ian/b.txt"),"Hello JDK7!".getBytes());

你也可以按照行写入文件,Files.write方法的参数中支持传递一个实现Iterable接口的类实例。将内容追加到指定文件可以使用write方法的第三个参数OpenOption

Files.write(Paths.get("/home/ian/b.txt"),"Hello JDK7!".getBytes(),
StandardOpenOption.APPEND);

默认情况Files类中的所有方法都会使用UTF-8编码进行操作,当你不愿意这么干的时候可以传递Carset参数进去变更。

当然Files还有一些其他的常用方法:

InputStream ins = Files.newInputStream(path);
OutputStream ops = Files.newOutputStream(path);
Reader reader = Files.newBufferedReader(path);
Writer writer = Files.newBufferedWriter(path);

创建、移动、删除

创建文件、目录

// 这个是如果路径不存在的话 为true,则创建
if(!Files.exists(path)){
    Files.createFile(path);
    Files.createDirectory(path);
}

Files 还提供了一些方法让我们创建临时文件、临时目录:

Files.createTempFile(dir, prefix, suffix);
Files.createTempFile(prefix, suffix);
Files.createTempDirectory(dir, prefix);
Files.createTempDirectory(prefix);

这里的dir是一个Path对象,并且字符串prefixsuffix都可能为null。 例如调用Files.createTempFile(null,".txt")会返回一个类似/tmp/37864823749.txt的文件

读取一个目录下的文件请使用Files.list和Files.walk方法

复制、移动一个文件内容到某个路径

Files.copy(in, path);
Files.move(path, path);

删除一个文件

Files.delete(path);

小的改进

Java8 是一个较大改变的版本,包含了API和库方面的修正,它还对我们常用的API进行很多微小的调整,下面我会带你了解字符串、集合、注解等新方法。

字符串

使用过javaScript的人可能会知道当我们将一个数组中的元素组合起来变成字符串有一个方法join,例如 我们经常用到将数组中的字符串拼接成勇逗号分隔的一长串,这在Java中是要写for循环来完成的。

Java8中添加了join方法帮你搞定这一切:

String str = String.join(",","a","b","c");

第一个参数是分隔符,后面接收一个CharSequence类型的可变参数数组或一个Iterable

集合

集合改变中最大的当属Stream API,除此之外还有一些小的改动。

  • Map中的很多方法对并发访问十分重要,后面再说。
  • Iterator提供forEachRemaining将剩余的元素传递给一个函数
  • BitSet可以产生一个Stream对象

通用目标类型判断

Java8 对泛型参数的推断进行了增强。相信你对Java8 之前版本中的类型推断已经比较熟悉了。 比如,Collections中的方法EmptyList方法定义如下:

Static<T> List<T> emptyList()

emptyList 方法使用了泛型类型 T进行参数化。你可以像下面这样为该类型参数提供一个显式的类型进行函数调用:

List<Person> persions = Collections.<Person>emptyList();

不过编译器也可以推断泛型参数的类型,上面的代码和下面这段代码是等价的:

List<person> persions = Collections.emptyList();

我还是习惯于这样书写。

注解

Java8 在两个方面对注解机制进行了改进,分别为:

  • 可以定义重复注解
  • 可以为任何类型添加注解

重复注解

之前版本的Java禁止对同样的注解类型声明多次。由于这个原因,下面的第二句代码是无效的:

@interface Basic{
    String name();
}
@Basic(name="fix")
@Basic(name="todo")
class Person{}

我们之前可能会通过数组的做法绕过这一限制:

@interface Basic{
    String name();
}
@interface Basics{
    Basic[] value();
}
@Basics({@Basic(name="fix"),@Basic("todo")})
class Person{}

Book 类的嵌套注解相当难看。这就是Java8 想要从根本上移除这一限制的原因,去掉这一限制后,代码的可读性会好很多。现在,如果你的配置允许重复注解,你可以毫无顾虑地一次声明多个同一种类型的注解。

它目前还不是默认行为,你需要显式地要求进行重复注解。

创建一个重复注解

如果一个注解在设计之初就是可重复的,你可以直接使用它。但是,如果你提供的注解是为用户提供的,那么就需要做一些工作,说明该注解可以重复。下面是你需要执行的两个步骤:

  1. 将注解标记为@Repeatable
  2. 提供一个注解的容器下面的例子展示了如何将@Basic注解修改为可重复注解
@Repeatable(Basics.class)
@interface Basic{
    String name();
}
@Retention(RetentionPolicy.RUNTIME)
@interface Basics{
    Basic[] value();
}

完成了这样的定义之后,Person类可以通过多个@Basic注解进行注释,如下所示:

@Basic(name = "fix")
@Basic(name = "todo")
class Person{}

编译时, Person 会被认为使用了 @Basics( { @Basic(name="fix") , @Basic(name="todo")} ) 这样的形式进行了注解,所以,你可以把这种新的机制看成是一种语法糖, 它提供了程序员之前利用的惯用法类似的功能。为了确保与反射方法在行为上的一致性, 注解会被封装到一个容器中。 Java API中的getAnnotation(Class<T> annotationClass)方法会为注解元素返回类型为T的注解。 如果实际情况有多个类型为T的注解,该方法的返回到底是哪一个呢?

我们不希望一下子就陷入细节的魔咒,类Class提供了一个新的getAnnotationsByType方法, 它可以帮助我们更好地使用重复注解。比如,你可以像下面这样打印输出Person类的所有Basic注解:

返回一个由重复注解Basic组成的数组

public static void main(String[] args) {
    Basic[] basics = Person.class.getAnnotationsByType(Basic.class);
    Arrays.asList(basics).forEach(a -> { 
        System.out.println(a.name()); 
    }); 
}

Null 检查

使用Optional优化代码

public static String getUserNameByOptional(User user){
    Optional<String> userName = 			 Optional.ofNullable(user).map(User::getName);
    return userName.orElse(null);
}

usernull的时候我们设置UserName的值为null,否则返回getName的返回值,但此时不会抛出异常空指针。

在之前的代码片段中是我们最熟悉的命令式编程思维,写下的代码可以描述程序的执行逻辑,得到什么样的结果。

后面这种方式是函数式思维方式,在函数式的思维方式里,结果比过程更重要,不需要关注执行的细节。程序的具体执行由编译器来决定。

这种情况下提高程序的性能是一个不容易的事情。

我们再次了解下Optional中的一些使用方法

Optional方法

你可以通过静态工厂方法Optional.empty,创建一个空的Optional对象:

Optional<User> emptyUser = Optional.empty();

创建一个非空值的Optional

Optinal<User> userOptional = Optional.of(user);

如果user是一个null,这段代码会立即抛出一个NullPointerException,而不是等到你试图访问user的属性值时才返回一个错误。

可接受nullOptional

Optional<User> ofNullOptional = Optional.ofNullable(user);

使用静态工厂方法Optional.ofNullable,你可以创建一个允许null值得Optional对象。

如果user 是 null,那么得到的Optional对象就是个空对象,但不会让你导致空指针。

使用 map 从 Optional对象中提取和转换值

Optional<User> ofNullOptional = Optional.ofNullable(user);
Optional<String> userName = ofNullOptional.map(User::getName);

这种操作就像我们之前在操作Stream是一样的,获取的只是User中的一个属性。

默认行为及解引用Optional对象

我们决定采用orElse方法读取这个变量的值,使用这种方式你还可以定义一个默认值, 遭遇空的Optional变量时,默认值会作为该方法的调用返回值。 Optional类提供了多种方法读取 Optional实例中的变量值。

  • get()是这些方法中最简单但又最不安全的方法。如果变量存在,它直接返回封装的变量 值,否则就抛出一个NoSuchElementException异常。所以,除非你非常确定Optional 变量一定包含值,否则使用这个方法是个相当糟糕的主意。此外,这种方式即便相对于 嵌套式的null检查,也并未体现出多大的改进。

  • orElse(T other)是我们在代码清单10-5中使用的方法,正如之前提到的,它允许你在 Optional对象不包含值时提供一个默认值。

  • orElseGet(Supplier<? extends T> other)orElse方法的延迟调用版,Supplier 方法只有在Optional对象不含值时才执行调用。如果创建默认值是件耗时费力的工作, 你应该考虑采用这种方式(借此提升程序的性能),或者你需要非常确定某个方法仅在

  • Optional为空时才进行调用,也可以考虑该方式(这种情况有严格的限制条件)。 orElseThrow(Supplier<? extends X> exceptionSupplier)get方法非常类似, 它们遭遇Optional对象为空时都会抛出一个异常,但是使用orElseThrow你可以定制希 望抛出的异常类型。

  • ifPresent(Consumer<? super T>)让你能在变量值存在时执行一个作为参数传入的 方法,否则就不进行任何操作。

当前除了这些Optional类也具备一些和Stream类似的API,我们先看看Optional类方法:

用Optional封装可能为null的值

目前我们写的大部分Java代码都会使用返回NULL的方式来表示不存在值,比如Map中通过Key获取值, 当不存在该值会返回一个null。 但是,正如我们之前介绍的,大多数情况下,你可能希望这些方法能返回一个Optional对象。 你无法修改这些方法的签名,但是你很容易用Optional对这些方法的返回值进行封装。

我们接着用Map做例子,假设你有一个Map<String, Object>类型的map,访问由key的值时, 如果map中没有与key关联的值,该次调用就会返回一个null。

Object value = map.get("key");

使用Optional封装map的返回值,你可以对这段代码进行优化。要达到这个目的有两种方式: 你可以使用笨拙的if-then-else判断语句,毫无疑问这种方式会增加代码的复杂度; 或者你可以采用Optional.ofNullable方法

Optional<Object> value = Optional.ofNullable(map.get("key"));

每次你希望安全地对潜在为null的对象进行转换,将其替换为Optional对象时,都可以考虑使用这种方法。



  相关文章:


留言区:

TOP