這篇文章主要講解了“關(guān)于Java8的知識(shí)點(diǎn)有哪些”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“關(guān)于Java8的知識(shí)點(diǎn)有哪些”吧!
創(chuàng)新互聯(lián)公司是一家從事企業(yè)網(wǎng)站建設(shè)、成都網(wǎng)站建設(shè)、成都做網(wǎng)站、行業(yè)門戶網(wǎng)站建設(shè)、網(wǎng)頁(yè)設(shè)計(jì)制作的專業(yè)的建站公司,擁有經(jīng)驗(yàn)豐富的網(wǎng)站建設(shè)工程師和網(wǎng)頁(yè)設(shè)計(jì)人員,具備各種規(guī)模與類型網(wǎng)站建設(shè)的實(shí)力,在網(wǎng)站建設(shè)領(lǐng)域樹(shù)立了自己獨(dú)特的設(shè)計(jì)風(fēng)格。自公司成立以來(lái)曾獨(dú)立設(shè)計(jì)制作的站點(diǎn)千余家。
在了解一項(xiàng)新技術(shù)之前,我們需要了解我們?yōu)槭裁匆W(xué)習(xí)它以及它的優(yōu)點(diǎn),以下是我總結(jié)的:
Java8(又稱jdk1.8)是java語(yǔ)言開(kāi)發(fā)的一個(gè)主要版本,Java8是oracal公司于2014年3月發(fā)布,可以看成是自java5以來(lái)最具有革命性的版本。
新特性的優(yōu)點(diǎn):速度更快、代碼更少、便于并行、最大化減少空指針異常
函數(shù)式編程提供了一種更高層次的抽象化
排序:
ListrolesListSort = rolesList.stream().sorted(Comparator.comparing(RoleEntity::getCreateDate)).collect(Collectors.toList());
Consumer是一個(gè)函數(shù)式接口
參數(shù)是Consumer類型的,Consumer里面的泛型表示泛型的類型要么是Integer,要么是Integer的父類,super表示它及它上面的,也就是父類。
下面這段代碼是在Iterable接口里面的默認(rèn)方法,jdk8之后的新方法,默認(rèn)方法(默認(rèn)方法的引入很大程度上是為了保證向后兼容)
default void forEach(Consumer super T> action) { Objects.requireNonNull(action); for (T t : this) { action.accept(t); } }
關(guān)于Java8的新特性,我總結(jié)了以下6個(gè)方面,我們可以從以下6個(gè)方面進(jìn)行學(xué)習(xí)了解:
一、Lambda表達(dá)式
我的理解lambbda表達(dá)式其實(shí)是新的一套語(yǔ)法規(guī)則,主要是語(yǔ)法上面的要求。
那我們?yōu)樯缎枰狶ambda表達(dá)式?
在java中,我們無(wú)法將函數(shù)作為參數(shù)傳遞給一個(gè)方法,也無(wú)法聲明返回一個(gè)函數(shù)的方法;在JavaScript中,函數(shù)參數(shù)是一個(gè)函數(shù),返回值是另一個(gè)函數(shù)的情況是非常常見(jiàn)的;JavaScript是一門非常典型的函數(shù)式語(yǔ)言。
addUser(e -> Sysout.out.println("hello"))e表示參數(shù),->箭頭符號(hào),表示分隔符,他的作用是分割左邊和右邊的。Sysout.out.println("hello")是執(zhí)行體,也就是代碼塊(如果執(zhí)行體里面不止一行代碼,那就可以加上花括號(hào)括起來(lái))所以Lambda表達(dá)式分為三部分
Lambda表達(dá)式的基本結(jié)構(gòu):
一個(gè)Lambda表達(dá)式可以有0個(gè)或多個(gè)參數(shù),參數(shù)的類型可以明確聲明,也可以通過(guò)上下文來(lái)推斷。例如(int a)和(a)效果一樣;
所有參數(shù)都必須包含在圓括號(hào)內(nèi),參數(shù)之間用逗號(hào)相隔;
空?qǐng)A括號(hào)代表參數(shù)集為空。例如:()-> 42
當(dāng)只有一個(gè)參數(shù),且其類型可以推導(dǎo)出時(shí),圓括號(hào)()可以省略。例如:a -> return a*a
Lambda表達(dá)式的主體也就是body可以包含0條或多條語(yǔ)句。
如果表達(dá)式的主體只有一條語(yǔ)句,花括號(hào){}可以省略,匿名函數(shù)的返回類型與該主體表達(dá)式一致
如果表達(dá)式的主體包含一條語(yǔ)句以上,則必須包含在花括號(hào){}里面形成代碼塊。匿名函數(shù)的返回類型與該主體表達(dá)式一致,若沒(méi)有返回則為空。
statement和expression的區(qū)別,expression只有一句,不需要花括號(hào)包裹,不需要return;statement需要花括號(hào)包裹,且如果有返回值,必須return
(argument)-> {body}
也可以:
(arg1, arg2)-> {body} (type arg1, type arg2)-> {body}(這個(gè)是最完整的語(yǔ)法) (param1,param2,param3)-> {} 左邊圓括號(hào)里面表示方法的參數(shù) ,右邊花括號(hào)里面代表方法的具體實(shí)現(xiàn) ()-> {} 類型是通過(guò)上下文來(lái)推斷的
實(shí)際就是去目標(biāo)函數(shù)式接口里面去找那個(gè)特定的唯一的抽象方法,去看抽象方法里面的-參數(shù)和返回類型,而抽象方法的名字對(duì)于Lambda表達(dá)式來(lái)說(shuō)是毫無(wú)意義的
Lambda表達(dá)式的作用:
Lambda表達(dá)式為Java添加了缺失的函數(shù)式編程特性,使我們能將函數(shù)當(dāng)作一等公民看待
在將函數(shù)作為一等公民的語(yǔ)言中,Lambda表達(dá)式的類型是函數(shù)。但在Java中,Lambda表達(dá)式是對(duì)象,他們必須依附于一類特別的對(duì)象類型——函數(shù)式接口(functional interface)
傳遞的是行為,而不僅僅是值(在以前的方式中,是先定義好了行為(行為已經(jīng)存在),然后再調(diào)用這個(gè)行為進(jìn)行使用,而現(xiàn)在是相反,行為是提前并不存在,是通過(guò)方法的傳遞來(lái)進(jìn)行告知的)
//內(nèi)部迭代 integerList.forEach(new Consumer() { //匿名內(nèi)部類 @Override public void accept(Integer integer) { System.out.println(integer); } });
二、函數(shù)式(Functional)接口
函數(shù)式接口是可以通過(guò)三種方式實(shí)現(xiàn)的:Lambda表達(dá)式、方法引用、構(gòu)造器引用
通過(guò)Lambda表達(dá)式、方法引用或者構(gòu)造器引用的來(lái)創(chuàng)建一個(gè)函數(shù)式接口的實(shí)例
關(guān)于函數(shù)式接口:
鴻蒙官方戰(zhàn)略合作共建——HarmonyOS技術(shù)社區(qū)
如果一個(gè)接口只有一個(gè)抽象方法,那么該接口就是一個(gè)函數(shù)式接口
如果我們?cè)谀硞€(gè)接口上聲明了@FunctionalInterface注解,那么編譯器就會(huì)按照函數(shù)式接口的定義來(lái)要求該接口。
如果一個(gè)接口只有一個(gè)抽象方法,但是在該接口上并沒(méi)有聲明@FunctionalInterface注解,那么編譯器依舊會(huì)把該接口看作一個(gè)函數(shù)式接口
Java8里面引入的很多函數(shù)式接口它們都位于java.util.function下面。
以下是一些常用的函數(shù)式接口:
位于java.util.function這個(gè)包下面
Consumer消費(fèi)者 接受一個(gè)參數(shù),不返回結(jié)果
public interface Consumer { void accept(T t); }
Function,接受一個(gè)參數(shù),返回一個(gè)結(jié)果
public interface Function { R apply(T t); }
BiFunction接收兩個(gè)參數(shù),返回一個(gè)結(jié)果(其中BI是bidirectional的縮寫(xiě),意思是雙向)
public interface BiFunction { R apply(T t, U u); }
Supplier 提供者,供應(yīng)者,不接收任何參數(shù),返回一個(gè)結(jié)果
public interface Supplier { T get(); }
Predicate謂語(yǔ),接收一個(gè)參數(shù),返回一個(gè)布爾值(根據(jù)給定的參數(shù),返回布爾)
public interface Predicate { boolean test(T t); }
三、方法引用
方法引用是Lambda表達(dá)式的一種特殊情況(或者說(shuō)是Lambda表達(dá)式的一個(gè)語(yǔ)法糖),可以理解為方法引用和Lambda表達(dá)式這兩種方式所實(shí)現(xiàn)的功能其實(shí)一樣的,完全等價(jià),但是方法引用的方式更簡(jiǎn)潔。
我們可以將方法引用看作是一個(gè)函數(shù)指針(Function pointer)
方法引用(method references):
ListintegerList = Arrays.asList(1,2,3,4,5); //方法引用的方式 integerList.forEach(System.out::println);
方法引用有4種:
1、類名::靜態(tài)方法名
以下這兩種形式是完全不等價(jià)的
classname::staticmethod(表示的是指向,函數(shù)指針的概念)
classname.staticmethod(真正表示的是方法調(diào)用的概念)
2、引用名(對(duì)象名)::實(shí)例方法名
3、類名::實(shí)例方法名
4、構(gòu)造方法引用(constructor references):類名::new
四、強(qiáng)大的Stream API
其實(shí)就是JDK8提供給我們新的API,經(jīng)常和Lambda表達(dá)式和函數(shù)式接口一起使用
分為串行流和并行流
list.stream()串行流,只有一個(gè)線程,一個(gè)線程執(zhí)行所有操作
list.parallelStream()并行流,多線程,分工合作
list.stream().map():map此處的意思是映射的意思
Stream也是一個(gè)接口,里面的絕大多數(shù)方法都是高階函數(shù)
Stream流,他是與Lambda表達(dá)式相伴相生的,通過(guò)流的方式我們可以更好的操作集合
流的三部分構(gòu)成:(SQL語(yǔ)句和流非常非常像)
1、源
2、零個(gè)或若干個(gè)中間操作(操作的是這個(gè)源,操作值的是過(guò)濾,排序,映射,分區(qū)等,這些操作本身有點(diǎn)像SQL語(yǔ)句)
3、終止操作
流操作分類:
1、惰性求值
2、及早求值
流的所有的中間操作方法都是lazy的(或者說(shuō)是延遲的,或者說(shuō)是惰性求值的),在沒(méi)有遇到終止操作或者及早求值的操作的情況下,中間操作是不會(huì)被執(zhí)行的,只有在遇到終止操作的時(shí)候,這若干個(gè)中間操作才會(huì)一并的執(zhí)行
stream().xxx().zzz().count();
filter()用來(lái)判斷里面的條件是真還是假?如果是假,就從流當(dāng)中過(guò)濾掉;如果是真,就繼續(xù)放到流當(dāng)中,供后續(xù)操作使用
流:
Colletion提供了新的Stream()方法;
流不存儲(chǔ)值,通過(guò)管道的方式獲取值;
本質(zhì)是函數(shù)式的,對(duì)流的操作會(huì)造成一個(gè)結(jié)果,不過(guò)并不會(huì)修改底層的數(shù)據(jù)源,集合可以作為流的底層數(shù)據(jù)源;
延遲查找,很多流操作(過(guò)濾,映射,排序,分區(qū)等)都可以延遲實(shí)現(xiàn);
SQL語(yǔ)句是一種描述性的語(yǔ)言,只需要發(fā)送指令告訴底層需要做什么,而不關(guān)心底層是怎么實(shí)現(xiàn)的,而流其實(shí)也是一樣的,只需要知道做什么,而不需要知道具體底層是怎么做的。
內(nèi)部迭代和外部迭代本質(zhì)刨析:(操作流就像英語(yǔ)中的完形填空,直接操作集合就是完成一個(gè)完整的命題作文)
內(nèi)部迭代
用流,是并行化,以下代碼可能你覺(jué)得有多個(gè)循環(huán),但是流的底層實(shí)際上只用了一個(gè)循環(huán),可以這樣想,流實(shí)際上是一個(gè)容器,里面有一個(gè)集合,這個(gè)集合存放的是對(duì)流的各種操作,流會(huì)盡最大可能去優(yōu)化;以下代碼也不是按照順序一個(gè)一個(gè)執(zhí)行的,是由集合框架自己決定的
外部迭代
用集合,是串行化,下圖是我的代碼,可以幫助大家理解
集合關(guān)注的是數(shù)據(jù)與數(shù)據(jù)存儲(chǔ)本身;
流關(guān)注的是對(duì)數(shù)據(jù)的計(jì)算;
流與迭代器類似的一點(diǎn)是:流是無(wú)法重復(fù)使用或消費(fèi)的
如何判斷是中間操作還是終止操作呢
中間操作都會(huì)返回一個(gè)Stream對(duì)象,比如Stream,Stream,Stream
終止操作則不會(huì)返回Steam類型,可能不返回值,也可能返回其他類型的單個(gè)值
Stream流里面的方法:
int sum = Stream.iterate(1, item -> item + 2).limit(6).filter(item -> item > 2) .mapToInt(item -> item * 2) .skip(2).limit(2).sum();
skip():忽略掉前幾個(gè)元素
limit():獲取前幾個(gè)元素
sum():求和(map映射是沒(méi)有求和方法的)
Stream分組與分區(qū)(partition ):
分組:group by
分區(qū):partition by (布爾值)
分區(qū)是分組的一種特殊情況
流的特性:
流一旦被操作或使用了,就不能再去重復(fù)的使用這個(gè)流,或者說(shuō)流一旦被關(guān)閉了,也是不能再去重復(fù)使用了
五、Optional類
中文意思:可選
Optional類的使用其實(shí)在其他語(yǔ)言里很早就使用了(比如Swift、Groovy、Scala),Java是最晚使用的,
它的出現(xiàn)主要解決的問(wèn)題:NPE(NullPointerException)
if (null != person){ Address address = person.getName(); if (null != address){ } }
六、高階函數(shù)
高階函數(shù):如果一個(gè)函數(shù)接受一個(gè)函數(shù)作為參數(shù),或者返回一個(gè)函數(shù)作為一個(gè)返回值,那么該函數(shù)就叫做高階函數(shù)。
默認(rèn)方法
接口當(dāng)中可以聲明方法的實(shí)現(xiàn)了,但是這個(gè)方法的實(shí)現(xiàn)必須要帶上default關(guān)鍵字
從java8開(kāi)始,為啥要增加默認(rèn)方法?
Collector收集器(很重要)
R collect(Collector collector);
collect:收集器
Collector作為collect方法的參數(shù)
Collector是一個(gè)接口,它是一個(gè)可變的匯聚操作,將輸入元素累積到一個(gè)可變的結(jié)果容器中(ArrayList就是一個(gè)可變的容器),它會(huì)在所有元素處理完畢之后,將累積的結(jié)果轉(zhuǎn)換成一個(gè)最終的表示(這是一個(gè)可選操作),它支持串行(一個(gè)線程執(zhí)行)和并行(多個(gè)線程執(zhí)行)兩種方式執(zhí)行。
Collectors本身提供了關(guān)于Collector的常見(jiàn)匯聚實(shí)現(xiàn),Collectors本身實(shí)際是一個(gè)工廠(Collectors提供了很多可變匯聚操作的實(shí)現(xiàn))
public interface Collector{ Supplier supplier(); BiConsumer accumulator();//翻譯成累加器 //將兩個(gè)結(jié)果容器合并成一個(gè)(用于線程并發(fā)) BinaryOperator combiner();//結(jié)合器 Function finisher();//完成器 }
Collector同一性和結(jié)合性分析
combiner函數(shù):
Iterator迭代器
感謝各位的閱讀,以上就是“關(guān)于Java8的知識(shí)點(diǎn)有哪些”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)關(guān)于Java8的知識(shí)點(diǎn)有哪些這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!