前言
上一篇咱們了解了一下內容:
① 方法引用
② 構造器引用
咱們繼續了解lambda表達式的剩下內容。
1.6 變量作用域
通常,我們希望能夠在lambda表達式的閉合方法或類中訪問其他的變量,例如:
public static void repeatMessage(String text,int count){
Runnable r=()->{
for(int i=0;i<count;i++){
Sysout.out.println(text);
Thread.yield();
}
};
new Thead(r).start();
}
假設有以下調用:
repeatMesage("Hello",1000);//在另一個線程中打印1000遍
注意看lambda表達式中count和text并沒有定義在lambda表達式中(它們是自由變量),而是方法repeatMessage的參數變量。
注意:含有自由變量的代碼塊被稱為“閉包(closure)”。內部類也會捕獲閉合域中的值。在Java 8之前,內部類只允許訪問final的局部變量。為了適應lambda表達式,這條規則也被放寬了。一個內部類可以訪問任何有效的final局部變量---即任何值不會發生變化的變量。
lambda表達式的方法體與嵌套代碼塊有著相同的作用域,因此它也適用同樣的命名沖突和屏蔽規則。在lambda表達式中不允許聲明一個與局部變量同名的參數或局部變量。
Path first=Paths.get("/usr/bin");
Comparator<String>comp=(first,second)->Integer.compare(first.length(),second.length());
//錯誤:變量first已經被定義。
在一個方法中,你不能有兩個同名的局部變量。因此,你也不能再lambda表達式中引入這樣的變量。
1.7 默認方法
接口中包含帶有具體實現的方法稱之為默認方法。
如果一個接口中定義了一個默認方法,而另外一個父類或接口中又定義了一個同名方法。該選擇哪個呢?根據以下原則選擇:
① 選擇父類中的方法。如果一個父類提供了具體實現方法,那么接口中具有相同名稱和參數的默認方法會被忽略。
② 接口沖突。如果一個父接口提供一個默認方法,而另一個接口也提供了具有相同名稱和參數類型的方法(不管該方法是否是默認方法),那么你必須通過覆蓋該方法來解決沖突。
1.8 接口中的靜態方法
在Java 8中你可以為接口提供靜態方法了。