Java1.8新特性

一、Lambda表達式和函數式接口

Lambda表達式(也稱爲閉包)是Java 8中最大和最使人期待的語言改變。它容許咱們將函數當成參數傳遞給某個方法,或者把代碼自己看成數據處理:函數式開發者很是熟悉這些概念。不少JVM平臺上的語言(Groovy、Scala等)從誕生之日就支持Lambda表達式,可是Java開發者沒有選擇,只能使用匿名內部類代替Lambda表達式。php

Lambda的設計耗費了不少時間和很大的社區力量,最終找到一種折中的實現方案,能夠實現簡潔而緊湊的語言結構。最簡單的Lambda表達式可由逗號分隔的參數列表、->符號和語句塊組成,例如:css

Arrays.asList( "a", "b", "d" ).forEach( e -> System.out.println( e ) );

二、接口的默認方法和靜態方法

 

Java 8使用兩個新概念擴展了接口的含義:默認方法和靜態方法。默認方法使得接口有點相似traits,不過要實現的目標不同。默認方法使得開發者能夠在 不破壞二進制兼容性的前提下,往現存接口中添加新的方法,即不強制那些實現了該接口的類也同時實現這個新加的方法。html

 

默認方法和抽象方法之間的區別在於抽象方法須要實現,而默認方法不須要。接口提供的默認方法會被接口的實現類繼承或者覆寫,例子代碼以下:java

 

private interface Defaulable { // Interfaces now allow default methods, the implementer may or // may not implement (override) them. default String notRequired() { return "Default implementation"; } } private static class DefaultableImpl implements Defaulable { } private static class OverridableImpl implements Defaulable { @Override public String notRequired() { return "Overridden implementation"; } }

 

Defaulable接口使用關鍵字default定義了一個默認方法notRequired()DefaultableImpl類實現了這個接口,同時默認繼承了這個接口中的默認方法;OverridableImpl類也實現了這個接口,但覆寫了該接口的默認方法,並提供了一個不一樣的實現。express

 

Java 8帶來的另外一個有趣的特性是在接口中能夠定義靜態方法,例子代碼以下:閉包

 

private interface DefaulableFactory { // Interfaces now allow static methods static Defaulable create( Supplier< Defaulable > supplier ) { return supplier.get(); } }

 

下面的代碼片斷整合了默認方法和靜態方法的使用場景:ide

 

public static void main( String[] args ) { Defaulable defaulable = DefaulableFactory.create( DefaultableImpl::new ); System.out.println( defaulable.notRequired() ); defaulable = DefaulableFactory.create( OverridableImpl::new ); System.out.println( defaulable.notRequired() ); }

 

這段代碼的輸出結果以下:函數

 

Default implementation Overridden implementation

 

因爲JVM上的默認方法的實如今字節碼層面提供了支持,所以效率很是高。默認方法容許在不打破現有繼承體系的基礎上改進接口。該特性在官方庫中的應用是:給java.util.Collection接口添加新方法,如stream()parallelStream()forEach()removeIf()等等。ui

儘管默認方法有這麼多好處,但在實際開發中應該謹慎使用:在複雜的繼承體系中,默認方法可能引發歧義和編譯錯誤。this

 

3 、方法引用

方法引用使得開發者能夠直接引用現存的方法、Java類的構造方法或者實例對象。方法引用和Lambda表達式配合使用,使得java類的構造方法看起來緊湊而簡潔,沒有不少複雜的模板代碼。

西門的例子中,Car類是不一樣方法引用的例子,能夠幫助讀者區分四種類型的方法引用。

public static class Car { public static Car create( final Supplier< Car > supplier ) { return supplier.get(); } public static void collide( final Car car ) { System.out.println( "Collided " + car.toString() ); } public void follow( final Car another ) { System.out.println( "Following the " + another.toString() ); } public void repair() { System.out.println( "Repaired " + this.toString() ); } }

第一種方法引用的類型是構造器引用,語法是Class::new,或者更通常的形式:Class<T>::new。注意:這個構造器沒有參數。

final Car car = Car.create( Car::new ); final List< Car > cars = Arrays.asList( car );

第二種方法引用的類型是靜態方法引用,語法是Class::static_method。注意:這個方法接受一個Car類型的參數。

cars.forEach( Car::collide );

第三種方法引用的類型是某個類的成員方法的引用,語法是Class::method,注意,這個方法沒有定義入參:

cars.forEach( Car::repair );

第四種方法引用的類型是某個實例對象的成員方法的引用,語法是instance::method。注意:這個方法接受一個Car類型的參數:

final Car police = Car.create( Car::new ); cars.forEach( police::follow );

運行上述例子,能夠在控制檯看到以下輸出(Car實例可能不一樣):

Collided com.javacodegeeks.java8.method.references.MethodReferences$Car@7a81197d Repaired com.javacodegeeks.java8.method.references.MethodReferences$Car@7a81197d Following the com.javacodegeeks.java8.method.references.MethodReferences$Car@7a81197d
相關文章
相關標籤/搜索