一提到Java 8就只能聽到lambda,但這不過是其中的一個而已,Java 8還有許多新的特性,有一些功能強大的新類別或者新的用法,還有一些功能則是早就該加到Java裡了,所以下面這篇文章主要來為大家介紹了關於Java8中大家可能忽略了的一些新特性,需要的朋友可以參考下。
前言
我們之前已經介紹了關於java8中lambda和函數式程式設計的相關內容,雖然我們開始了Java8的旅程,但是很多人直接從java6上手了java8, 也許有一些JDK7的特性你還不知道,在本章節中帶你回顧一下我們忘了的那些特性。 儘管我們不能講所有特性都講一遍,挑出常用的核心特性拎出來一起學習。
異常改進
#try-with-resources
這個特性是在JDK7種出現的,我們在之前操作一個流對象的時候大概是這樣的:
try { // 使用流对象 stream.read(); stream.write(); } catch(Exception e){ // 处理异常 } finally { // 关闭流资源 if(stream != null){ stream.close(); } }
這樣無疑有些繁瑣,而且finally塊還有可能拋出異常。在JDK7種提出了try-with-resources機制, 它規定你操作的類別只要是實作了AutoCloseable介面就可以在try語句區塊退出的時候自動呼叫close 方法關閉串流資源。
public static void tryWithResources() throws IOException { try( InputStream ins = new FileInputStream("/home/biezhi/a.txt") ){ char charStr = (char) ins.read(); System.out.print(charStr); } }
使用多個資源
try ( InputStream is = new FileInputStream("/home/biezhi/a.txt"); OutputStream os = new FileOutputStream("/home/biezhi/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 e) { e.printStackTrace(); }
這樣程式碼寫起來要捕獲很多異常,不是很優雅,JDK7種允許你捕獲多個異常:
try { Thread.sleep(20000); FileInputStream fis = new FileInputStream("/a/b.txt"); } catch (InterruptedException | IOException e) { e.printStackTrace(); }
且catch語句後面的例外參數是final的,不可以再修改/複製。
處理反射例外
使用過反射的同學可能知道我們有時候操作反射方法的時候會拋出很多不相關的檢查異常,例如:
try { Class<?> clazz = Class.forName("com.biezhi.apple.User"); clazz.getMethods()[0].invoke(object); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); }
儘管你可以使用catch多個異常的方法將上述異常都捕獲,但這也讓人感到痛苦。 JDK7修復了這個缺陷,引入了一個新類別ReflectiveOperationException可以幫你捕獲這些反射異常:
try { Class<?> clazz = Class.forName("com.biezhi.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 (line != null) { sb.append(line); sb.append(System.lineSeparator()); line = br.readLine(); } String everything = sb.toString(); } catch (Exception e){ e.printStackTrace(); } finally { try { br.close(); } catch (IOException e) { e.printStackTrace(); } }
大家對這樣的一段程式碼一定不陌生,但這樣太繁瑣了,我只想讀取一個文字文件,要寫這麼多程式碼還要處理讓人頭大的一堆異常,怪不得別人吐槽Java臃腫,是在下輸了。 。 。
下面我要介紹在JDK7中是如何改善這些問題的。
Path
Path用於表示文件路徑和文件,和File物件類似,Path物件並不一定要對應一個實際存在的文件, 它只是一個路徑的抽象序列。
要建立一個Path物件有多種方法,首先是final類別Paths的兩個static方法,如何從一個路徑字串來建構Path物件:
Path path1 = Paths.get("/home/biezhi", "a.txt"); Path path2 = Paths.get("/home/biezhi/a.txt"); URI u = URI.create("file:////home/biezhi/a.txt"); Path pathURI = Paths.get(u);
透過FileSystems建構
Path filePath = FileSystems.getDefault().getPath("/home/biezhi", "a.txt");
Path、URI、File之間的轉換
File file = new File("/home/biezhi/a.txt"); Path p1 = file.toPath(); p1.toFile(); file.toURI();
讀寫檔案
你可以使用Files類別快速實現檔案操作,例如讀取檔案內容:
byte[] data = Files.readAllBytes(Paths.get("/home/biezhi/a.txt")); String content = new String(data, StandardCharsets.UTF_8);
如果希望按照行讀取文件,可以呼叫
List<String> lines = Files.readAllLines(Paths.get("/home/biezhi/a.txt"));
反之你想將字串寫入到文件可以呼叫
Files.write(Paths.get("/home/biezhi/b.txt"), "Hello JDK7!".getBytes());
你也可以依照行寫入文件,Files.write方法的參數中支援傳遞一個實作Iterable介面的類別實例。 將內容追加到指定檔案可以使用write方法的第三個參數OpenOption:
Files.write(Paths.get("/home/biezhi/b.txt"), "Hello JDK7!".getBytes(), StandardOpenOption.APPEND);
預設情況Files類別中的所有方法都會使用UTF-8編碼進行操作,當你不願意這麼幹的時候可以傳遞Charset參數進去變更。
當然Files還有一些其他的常用方法:
InputStream ins = Files.newInputStream(path); OutputStream ops = Files.newOutputStream(path); Reader reader = Files.newBufferedReader(path); Writer writer = Files.newBufferedWriter(path);
建立、移動、刪除
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对象,并且字符串prefix和suffix都可能为null。 例如调用Files.createTempFile(null, ".txt")
会返回一个类似/tmp/21238719283331124678.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,除此之外还有一些小的改动。
类/接口 | 新方法 |
---|---|
Iterable | foreach |
Collection | removeIf |
List | replaceAll, sort |
Map | forEach, replace, replaceAll, remove(key, value), putIfAbsent, compute, computeIf, merge |
Iterator | forEachRemaining |
BitSet | stream |
Map中的很多方法对并发访问十分重要,我们将在后面的章节中介绍
Iterator提供forEachRemaining将剩余的元素传递给一个函数
BitSet可以产生一个Stream对象
通用目标类型判断
Java8对泛型参数的推断进行了增强。相信你对Java8之前版本中的类型推断已经比较熟悉了。 比如,Collections中的方法emptyList方法定义如下:
static <T> List<T> emptyList();
emptyList方法使用了类型参数T进行参数化。 你可以像下面这样为该类型参数提供一个显式的类型进行函数调用:
List<Person> persons = Collections.<Person>emptyList();
不过编译器也可以推断泛型参数的类型,上面的代码和下面这段代码是等价的:
List<Person> persons = Collections.emptyList();
我还是习惯于这样书写。
注解
Java 8在两个方面对注解机制进行了改进,分别为:
可以定义重复注解
可以为任何类型添加注解
重复注解
之前版本的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(name="todo") } ) class Person{ }
Book类的嵌套注解相当难看。这就是Java8想要从根本上移除这一限制的原因,去掉这一限制后, 代码的可读性会好很多。现在,如果你的配置允许重复注解,你可以毫无顾虑地一次声明多个同一种类型的注解。 它目前还不是默认行为,你需要显式地要求进行重复注解。
创建一个重复注解
如果一个注解在设计之初就是可重复的,你可以直接使用它。但是,如果你提供的注解是为用户提供的, 那么就需要做一些工作,说明该注解可以重复。下面是你需要执行的两个步骤:
将注解标记为@Repeatable
提供一个注解的容器下面的例子展示了如何将@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(Class8742468051c85b06f0a0af9e3e506b5c 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检查
Objects类添加了两个静态方法isNull和nonNull,在使用流的时候非常有用。
例如获取一个流的所有不为null的对象:
Stream.of("a", "c", null, "d") .filter(Objects::nonNull) .forEach(System.out::println);
Optional
空指针异常一直是困扰Java程序员的问题,也是我们必须要考虑的。当业务代码中充满了if else判断null 的时候程序变得不再优雅,在Java8中提供了Optional类为我们解决NullPointerException。
我们先来看看这段代码有什么问题?
class User { String name; public String getName() { return name; } } public static String getUserName(User user){ return user.getName(); }
这段代码看起来很正常,每个User都会有一个名字。所以调用getUserName方法会发生什么呢? 实际这是不健壮的程序代码,当User对象为null的时候会抛出一个空指针异常。
我们普遍的做法是通过判断user != null然后获取名称
public static String getUserName(User user){ if(user != null){ return user.getName(); } return null; }
但是如果对象嵌套的层次比较深的时候这样的判断我们需要编写多少次呢?难以想象
处理空指针
使用Optional优化代码
public static String getUserNameByOptional(User user) { Optional<String> userName = Optional.ofNullable(user).map(User::getName); return userName.orElse(null); }
当user为null的时候我们设置UserName的值为null,否则返回getName的返回值,但此时不会抛出空指针。
在之前的代码片段中是我们最熟悉的命令式编程思维,写下的代码可以描述程序的执行逻辑,得到什么样的结果。 后面的这种方式是函数式思维方式,在函数式的思维方式里,结果比过程更重要,不需要关注执行的细节。程序的具体执行由编译器来决定。 这种情况下提高程序的性能是一个不容易的事情。
我们再次了解下Optional中的一些使用方法
Optional方法
创建 Optional 对象
你可以通过静态工厂方法Optional.empty,创建一个空的Optional对象:
Optional<User> emptyUser = Optional.empty();
创建一个非空值的Optional
Optional<User> userOptional = Optional.of(user);
如果user是一个null,这段代码会立即抛出一个NullPointerException,而不是等到你试图访问user的属性值时才返回一个错误。
可接受null的Optional
Optional<User> ofNullOptional = Optional.ofNullable(user);
使用静态工厂方法Optional.ofNullable
,你可以创建一个允许null值的Optional对象。
如果user是null,那么得到的Optional对象就是个空对象,但不会让你导致空指针。
使用map从Optional对象中提取和转换值
Optional<User> ofNullOptional = Optional.ofNullable(user); OptionaluserName = ofNullOptional.map(User::getName);
这种操作就像我们之前在操作Stream是一样的,获取的只是User中的一个属性。
默认行为及解引用Optional对象
我们决定采用orElse方法读取这个变量的值,使用这种方式你还可以定义一个默认值, 遭遇空的Optional变量时,默认值会作为该方法的调用返回值。 Optional类提供了多种方法读取 Optional实例中的变量值。
get()
是这些方法中最简单但又最不安全的方法。如果变量存在,它直接返回封装的变量 值,否则就抛出一个NoSuchElementException异常。所以,除非你非常确定Optional 变量一定包含值,否则使用这个方法是个相当糟糕的主意。此外,这种方式即便相对于 嵌套式的null检查,也并未体现出多大的改进。
orElse(T other)
是我们在代码清单10-5中使用的方法,正如之前提到的,它允许你在 Optional对象不包含值时提供一个默认值。
orElseGet(Supplierd203bb1ae585225d4838a2b7e3d0503e other)
是orElse方法的延迟调用版,Supplier 方法只有在Optional对象不含值时才执行调用。如果创建默认值是件耗时费力的工作, 你应该考虑采用这种方式(借此提升程序的性能),或者你需要非常确定某个方法仅在 Optional为空时才进行调用,也可以考虑该方式(这种情况有严格的限制条件)。
orElseThrow(Supplierb4690ad92a9d39463cecfa62549165e4 exceptionSupplier)
和get方法非常类似, 它们遭遇Optional对象为空时都会抛出一个异常,但是使用orElseThrow你可以定制希 望抛出的异常类型。
ifPresent(Consumer117c5a0bdb71ea9a9d0c2b99b03abe3e)
让你能在变量值存在时执行一个作为参数传入的 方法,否则就不进行任何操作。
当前除了这些Optional类也具备一些和Stream类似的API,我们先看看Optional类方法:
方法 | 描述 |
---|---|
empty | 返回一个空的 Optional 实例 |
get | 如果该值存在,将该值用Optional包装返回,否则抛出一个NoSuchElementException异常 |
ifPresent | 如果值存在,就执行使用该值的方法调用,否则什么也不做 |
isPresent | 如果值存在就返回true,否则返回false |
filter | 如果值存在并且满足提供的谓词,就返回包含该值的 Optional 对象; 否则返回一个空的Optional对象 |
map | 如果值存在,就对该值执行提供的 mapping 函数调用 |
flatMap | 如果值存在,就对该值执行提供的 mapping 函数调用, 返回一个 Optional 类型的值,否则就返 回一个空的Optional对象 |
of | 将指定值用 Optional 封装之后返回,如果该值为null,则抛出一个NullPointerException异常 |
ofNullable | 将指定值用 Optional 封装之后返回,如果该值为 null,则返回一个空的Optional对象 |
orElse | 如果有值则将其返回,否则返回一个默认值 |
orElseGet | 如果有值则将其返回,否则返回一个由指定的 Supplier 接口生成的值 |
orElseThrow | 如果有值则将其返回,否则抛出一个由指定的 Supplier 接口生成的异常 |
用Optional封装可能为null的值
目前我们写的大部分Java代码都会使用返回NULL的方式来表示不存在值,比如Map中通过Key获取值, 当不存在该值会返回一个null。 但是,正如我们之前介绍的,大多数情况下,你可能希望这些方法能返回一个Optional对象。 你无法修改这些方法的签名,但是你很容易用Optional对这些方法的返回值进行封装。
我们接着用Map做例子,假设你有一个Map994a833a6ffa28d85b72cb15422c29d6
类型的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对象时,都可以考虑使用这种方法。
总结
以上是Java8中可能被忽略的新特性分享的詳細內容。更多資訊請關注PHP中文網其他相關文章!