google對(duì)java的支持是比較滯后的挠阁,一直到Api21才開(kāi)始支持java7。在java8推出兩年之后硫朦,google終于在Android N也就是Android7.x中支持java8了贷腕。作為商業(yè)項(xiàng)目,現(xiàn)在使用Android N顯然為時(shí)過(guò)早咬展,但是畢竟這是趨勢(shì)泽裳,早晚要用上的,需要提前學(xué)習(xí)破婆。
java8最大的亮點(diǎn)當(dāng)然是lambda表達(dá)式涮总,也許你認(rèn)為lambda表達(dá)式只是書(shū)寫(xiě)形式的改變,但是lambda表達(dá)式還牽扯到接口的靜態(tài)方法和default方法祷舀,牽扯到最近大火的rxjava的流式書(shū)寫(xiě)形式瀑梗,綜合起來(lái)就給我們帶來(lái)了重大利好。
lambda表達(dá)式基本:
首先說(shuō)下現(xiàn)在最常見(jiàn)的lambda表達(dá)式裳扯,在AndroidStudio中如果你使用了新版本的AndroidStudio(具體哪個(gè)版本忘記了抛丽,1.5?2.0饰豺?)亿鲜,AS會(huì)自動(dòng)幫你折疊成lambda表達(dá)式,相信你已經(jīng)見(jiàn)過(guò)不少這樣的:
view.setOnClickListener((v)->{ Sysout.out.println(v.getId()); });
展開(kāi)之后是這樣的:
view.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Sysout.out.println(v.getId());
}
});
當(dāng)然我們還可以更加簡(jiǎn)化為:
view.setOnclickListener((v)->Sysout.out.println(v.getId()));
再來(lái)看一個(gè):
以前我們的線程函數(shù)都是要寫(xiě)成這樣的:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello world !");
}
}).start();
用lambda可以這樣寫(xiě):
new Thread(() -> System.out.println("Hello world !")).start();
怎么樣冤吨,代碼是不是清爽了很多蒿柳。過(guò)濾掉了影響視線的接口和接口方法,只關(guān)注傳值和實(shí)現(xiàn)語(yǔ)句锅很,一眼就能抓住重點(diǎn)其馏,不管你愛(ài)不愛(ài),反正我是愛(ài)死了爆安。
如果你認(rèn)為lambda只用這點(diǎn)用處叛复,那lambda估計(jì)也不用混了。
有時(shí)候Lambda表達(dá)式的代碼就只是一個(gè)簡(jiǎn)單的方法調(diào)用而已,遇到這種情況褐奥,Lambda表達(dá)式還可以進(jìn)一步簡(jiǎn)化為 方法引用(Method References) 咖耘。
方法引用:
一共有四種形式的方法引用:
第一種引用 靜態(tài)方法 ,例如:
//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);
第二種引用 某個(gè)特定對(duì)象的實(shí)例方法:
//lambda
words.forEach((s)->System.out.println(s));
//Method Refrences
words.forEach(System.out::println);
第三種引用 某個(gè)類(lèi)的實(shí)例方法:
//lambda
words.stream().map(word -> word.length());
// method reference
words.stream().map(String::length);
第四種引用類(lèi)的 構(gòu)造函數(shù):
// lambda
words.stream().map(word -> { return new StringBuilder(word);});
// constructor reference
words.stream().map(StringBuilder::new);
結(jié)合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的文件轉(zhuǎn)換成圖像撬码,然后切換到主線程在view中顯示儿倒。關(guān)于RxJava的內(nèi)容后期我會(huì)單獨(dú)寫(xiě)一篇介紹的,或者也可以看RxJava詳解這篇文章呜笑。如果使用lambda就可以簡(jiǎn)化成:
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)
});
怎么樣夫否,如果你有代碼潔癖我相信你看完之后會(huì)立刻迫不及待的使用lambda了,如果現(xiàn)在Api24以下使用lambda叫胁,可以安裝retrolambda插件凰慈,但是就我個(gè)人而言我不太喜歡用,畢竟java8馬上就可以使用了驼鹅,并且使用了retrolambda僅僅是讓你使用lambda而已微谓,其他的java8特性是行不通的。
AndroidStudio 2.4 就開(kāi)始正常支持Java8了输钩,現(xiàn)在應(yīng)該只有預(yù)覽版豺型,等正式發(fā)布在用吧。這次使用就是真正的java8了买乃,我們可以使用foreach姻氨,也可已使用新的java.time的api了。