一伴澄、優(yōu)化點(diǎn)
- HashMap當(dāng)單鏈表數(shù)量大于8時(shí)轉(zhuǎn)為紅黑樹(shù)
- ConcurrentHashMap去掉了分段鎖的設(shè)計(jì),改為使用CAS實(shí)現(xiàn)
- 取消了JVM內(nèi)存中的方法區(qū),使用元空間替代(Matespace)腮郊,使用直接內(nèi)存
二、Lambad表達(dá)式
可以將一個(gè)方法作為參數(shù)進(jìn)行傳遞惊畏,可以理解為匿名內(nèi)部類的語(yǔ)法糖
基本語(yǔ)法:() -> {}
左側(cè):Lambad表達(dá)式的參數(shù)列表
右側(cè):Lambad表達(dá)式需要執(zhí)行的功能,既Lambad體Lambad表達(dá)式需要“函數(shù)式接口”的支持
函數(shù)式接口:接口中只有一個(gè)抽象方法的接口密任,稱為函數(shù)式接口颜启。可以使用注解@FunctionalInterface修飾浪讳,檢查是否是函數(shù)式接口缰盏。-
內(nèi)置4大核心函數(shù)式接口
- Consumer<T> 消費(fèi)型接口
void accept(T t); - Supplier<T> 供給型接口
T get(); - Function<T,R> 函數(shù)型接口
R apply(T t); - Predicate<T> 斷言型接口
boolean test(T t);
- Consumer<T> 消費(fèi)型接口
// 匿名內(nèi)部類的使用方式
new Thread(new Runnable() {
@Override
public void run() {
System.out.println(1);
}
}).start();
// Lambad表達(dá)式使用方式
new Thread(() -> System.out.println(1)).start();
三、方法引用
若Lambad體中的內(nèi)容有方法已經(jīng)實(shí)現(xiàn)了淹遵,我們可以使用“方法引用”直接引用該方法
使用前提:引用的方法參數(shù)列表和返回值類型必須與Lambad表達(dá)式一致
方法引用
- ObjectName::method(實(shí)例方法)
- ClassName::staticMethod(靜態(tài)方法)
- ClassName::method(如果第一個(gè)參數(shù)是實(shí)例方法的調(diào)用者口猜,第二個(gè)參數(shù)是實(shí)例方法的參數(shù)的情況下)
BiPredicate<String, String> bp = (x, y) -> x.equals(y); BiPredicate<String, String> bp2 = String::equals;
構(gòu)造器引用
- ClassName::new(參數(shù)列表能對(duì)應(yīng))
數(shù)組引用
- Type[]::new
// 使用示例
System.out::println
Integer::new
String[]::new
四、Stream API
Java 8 API添加了一個(gè)新的抽象稱為流Stream透揣,可以讓你以一種聲明的方式處理數(shù)據(jù)济炎。Stream 使用一種類似用 SQL 語(yǔ)句從數(shù)據(jù)庫(kù)查詢數(shù)據(jù)的直觀方式來(lái)提供一種對(duì) Java 集合運(yùn)算和表達(dá)的高階抽象。Stream API可以極大提高Java程序員的生產(chǎn)力辐真,讓程序員寫(xiě)出高效率须尚、干凈、簡(jiǎn)潔的代碼拆祈。這種風(fēng)格將要處理的元素集合看作一種流恨闪, 流在管道中傳輸倘感, 并且可以在管道的節(jié)點(diǎn)上進(jìn)行處理放坏, 比如篩選, 排序老玛,聚合等淤年。元素流在管道中經(jīng)過(guò)中間操作(intermediate operation)的處理钧敞,最后由最終操作(terminal operation)得到前面處理的結(jié)果。
注意:多個(gè)中間操作可以連接起來(lái)形成一個(gè)流水線麸粮,除非流水線上觸發(fā)終止操作溉苛,否則中間操作不會(huì)執(zhí)行任何的處理。在終止操作一次性全部處理弄诲,稱為“惰性求值”
針對(duì)大量數(shù)據(jù)的處理可以使用并行流愚战,使用 parallel() 切換并行流
操作步驟
-
創(chuàng)建Stream
- 通過(guò) Collection 系列集合提供的 stream() 或 parallelStream()方法獲取
List<String> list = new ArrayList<>(); Stream<String> stream = list.stream();
- 通過(guò) Arrays 中的靜態(tài)方法 stream() 獲取數(shù)組流
Stream<Integer> stream = Arrays.stream(new Integer[]{1,2,3,4});
- 通過(guò) Stream 中的靜態(tài)方法 of()
Stream<String> stream = Stream.of("hello","world","haha");
- 創(chuàng)建無(wú)限流
// 迭代 Stream<Integer> stream1 = Stream.iterate(0, x -> x + 2); // 生成 Stream<Double> stream2 = Stream.generate(() -> Math.random());
- 通過(guò) Collection 系列集合提供的 stream() 或 parallelStream()方法獲取
-
中間操作
- 篩選與切片
- filter() 接收Lambad,從流中排除某些元素
- limit() 截?cái)嗔鳎?使元素不超過(guò)給定數(shù)量
- skip(n) 跳過(guò)前 n 個(gè)元素齐遵,若不足 n 個(gè)寂玲,則返回空流
- distinct() 去重,通過(guò)流生成元素的 hashCode() 和 equals() 去除重復(fù)元素
- 映射
- map() 接收Lambad梗摇,將元素轉(zhuǎn)換成其他形式或提取信息拓哟。接收一個(gè)函數(shù)作為參數(shù),該函數(shù)會(huì)被應(yīng)用到每一個(gè)元素上伶授,并映射成一個(gè)新的元素
- flatMap() 接收一個(gè)Lambad断序,將流中的每個(gè)元素都轉(zhuǎn)換成另一個(gè)流,然后把所有流合成一個(gè)流
- 排序
- sorted() 自然排序
- sorted(Comparator comparator) 定制排序
- 篩選與切片
-
終止操作
- 查找與匹配
- allMatch() 檢查是否匹配所有元素
- anyMatch() 檢查是否至少匹配一個(gè)元素
- noneMatch() 檢查是否沒(méi)有匹配的元素
- findFirst() 返回第一個(gè)元素
- findAny() 返回當(dāng)前流中的任意元素
- count() 返回流中元素的總個(gè)數(shù)
- max() 返回流中的最大值
- min() 返回流中的最小值
- 歸約
- reduce(T identity, BinaryOperator operator) / reduce(BinaryOperator operator) 可以將流中的元素反復(fù)結(jié)合起來(lái)糜烹,得到一個(gè)值
- 收集
- collect() 將流轉(zhuǎn)換為其他形式违诗。接收一個(gè) Collector 接口的實(shí)現(xiàn),用于給 Stream 中元素做匯總的方法
Collector 接口中方法的實(shí)現(xiàn)決定了如何對(duì)流執(zhí)行收集操作(如收集到List疮蹦、Set较雕、Map)。但是 Collectors 實(shí)用類提供了很多靜態(tài)方法挚币,可以方便的創(chuàng)建常見(jiàn)收集器實(shí)例亮蒋。
- collect() 將流轉(zhuǎn)換為其他形式违诗。接收一個(gè) Collector 接口的實(shí)現(xiàn),用于給 Stream 中元素做匯總的方法
- 查找與匹配
五、Optional 類
Optional<T> 類(java.util.Optional)是一個(gè)容器類妆毕,代表一個(gè)值存在或不存在慎玖,原來(lái)用 null 表示一個(gè)值不存在,現(xiàn)在 Optional 可以更好的表達(dá)這個(gè)概念笛粘。并且可以避免空指針異常趁怔。
常用方法
- of(T t) 創(chuàng)建一個(gè)Optional實(shí)例
- empty() 創(chuàng)建一個(gè)空的Optional實(shí)例
- ofNullable(T t) 若 t 不為 null 創(chuàng)建Optional實(shí)例,否則創(chuàng)建空的Optional實(shí)例
- isPresent() 判斷是否包含值
- orElse(T t) 若容器包含值則返回該值薪前,否則返回 t
- orElseGet(Supplier s) 類似 orElse() ,只不過(guò)參數(shù)為一個(gè)獲取值的 Lambad 表達(dá)式
- map(Function f) 有值則對(duì)值進(jìn)行處理润努,無(wú)值則字節(jié)返回 empty()
- flatMap(Function f) 類似 map() ,要求返回值必須是 Optional
六、接口中的默認(rèn)方法和靜態(tài)方法
Java8開(kāi)始接口中允許存在默認(rèn)方法和靜態(tài)方法
注意:
1.當(dāng)一個(gè)類的父類和實(shí)現(xiàn)的接口中有相同的方法時(shí)示括,遵循“類優(yōu)先”原則
2.當(dāng)一個(gè)類實(shí)現(xiàn)的兩個(gè)接口中有相同的方法時(shí)铺浇,則類必須重寫(xiě)該方法
public interface MyInterface {
// 默認(rèn)方法
default String getName() {
return "111";
}
// 靜態(tài)方法
public static void test() {
System.out.println("222");
}
}
七、新的時(shí)間日期API
Java8提供了一套全新的時(shí)間日期API垛膝,是線程安全的鳍侣。
- LocalDate丁稀、LocalTime、LocalDateTime
- Instant 時(shí)間戳
- Duration 計(jì)算兩個(gè)時(shí)間之間的間隔
- Period 計(jì)算兩個(gè)日期之間的間隔
- TemporalAdjuster 時(shí)間校正器
- DateTimeFormatter 格式化時(shí)間/日期
- ZonedDate倚聚、ZonedTime线衫、ZonedDateTime 時(shí)區(qū)
八、重復(fù)注解與類型注解
重復(fù)注解就是指同一個(gè)注解在某個(gè)地方可以使用多次惑折,Java提供了一個(gè)元注解@Repeatable可以實(shí)現(xiàn)重復(fù)注解
// 定義一個(gè)注解授账,使用Repeatable指定注解的容器
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Repeatable(MyAnnotationContainer.class)
public @interface MyAnnotation {
String value();
}
// 注解MyAnnotation的容器
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotationContainer {
MyAnnotation[] value();
}
// 使用重復(fù)注解
@MyAnnotation("hello")
@MyAnnotation("world")
public void hello() {
System.out.println(1);
}