google對java的支持是比較滯后的,一直到Api21才開始支持java7。在java8推出兩年之后,google終于在Android N也就是Android7.x中支持java8了。作為商業項目,現在使用Android N顯然為時過早,但是畢竟這是趨勢,早晚要用上的,需要提前學習。
java8最大的亮點當然是lambda表達式,也許你認為lambda表達式只是書寫形式的改變,但是lambda表達式還牽扯到接口的靜態方法和default方法,牽扯到最近大火的rxjava的流式書寫形式,綜合起來就給我們帶來了重大利好。
lambda表達式基本:
首先說下現在最常見的lambda表達式,在AndroidStudio中如果你使用了新版本的AndroidStudio(具體哪個版本忘記了,1.5?2.0?),AS會自動幫你折疊成lambda表達式,相信你已經見過不少這樣的:
view.setOnClickListener((v)->{ Sysout.out.println(v.getId()); });
展開之后是這樣的:
view.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Sysout.out.println(v.getId());
}
});
當然我們還可以更加簡化為:
view.setOnclickListener((v)->Sysout.out.println(v.getId()));
再來看一個:
以前我們的線程函數都是要寫成這樣的:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello world !");
}
}).start();
用lambda可以這樣寫:
new Thread(() -> System.out.println("Hello world !")).start();
怎么樣,代碼是不是清爽了很多。過濾掉了影響視線的接口和接口方法,只關注傳值和實現語句,一眼就能抓住重點,不管你愛不愛,反正我是愛死了。
如果你認為lambda只用這點用處,那lambda估計也不用混了。
有時候Lambda表達式的代碼就只是一個簡單的方法調用而已,遇到這種情況,Lambda表達式還可以進一步簡化為 方法引用(Method References) 。
方法引用:
一共有四種形式的方法引用:
第一種引用 靜態方法 ,例如:
//lambda
List<Integer> ints = Arrays.asList(1, 2, 3);
ints.sort((i1, i2)->Integer.compare(i1, i2));
//Method References
List<Integer> ints = Arrays.asList(1, 2,3);
ints.sort(Integer::compare);
第二種引用 某個特定對象的實例方法:
//lambda
words.forEach((s)->System.out.println(s));
//Method Refrences
words.forEach(System.out::println);
第三種引用 某個類的實例方法:
//lambda
words.stream().map(word -> word.length());
// method reference
words.stream().map(String::length);
第四種引用類的 構造函數:
// lambda
words.stream().map(word -> { return new StringBuilder(word);});
// constructor reference
words.stream().map(StringBuilder::new);
結合RxJava的使用:
Observable.from(folders)
.flatMap(new Func1<File, Observable<File>>() {
@Override
public Observable<File> call(File file) {
return Observable.from(file.listFiles());
}
})
.filter(new Func1<File, Boolean>() {
@Override
public Boolean call(File file) {
return file.getName().endsWith(".png");
}
})
.map(new Func1<File, Bitmap>() {
@Override
public Bitmap call(File file) {
return getBitmapFromFile(file);
}
})
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Action1<Bitmap>() {
@Override
public void call(Bitmap bitmap) {
imageCollectorView.addImage(bitmap);
}
});
這一段的意思就是在io線程(RxJava定義的線程池的子線程)從文件夾中獲取后綴名為.png的文件轉換成圖像,然后切換到主線程在view中顯示。關于RxJava的內容后期我會單獨寫一篇介紹的,或者也可以看RxJava詳解這篇文章。如果使用lambda就可以簡化成:
Observable.from(folders)
.flatMap((Func1) (folder) -> {
Observable.from(file.listFiles())
})
.filter((Func1) (file) -> {
file.getName().endsWith(".png")
})
.map((Func1) (file) -> {
getBitmapFromFile(file)
})
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe((Action1) (bitmap) -> {
imageCollectorView.addImage(bitmap)
});
怎么樣,如果你有代碼潔癖我相信你看完之后會立刻迫不及待的使用lambda了,如果現在Api24以下使用lambda,可以安裝retrolambda插件,但是就我個人而言我不太喜歡用,畢竟java8馬上就可以使用了,并且使用了retrolambda僅僅是讓你使用lambda而已,其他的java8特性是行不通的。
AndroidStudio 2.4 就開始正常支持Java8了,現在應該只有預覽版,等正式發布在用吧。這次使用就是真正的java8了,我們可以使用foreach,也可已使用新的java.time的api了。