文章目錄
java8 新特性之 – lamdba 表達式
定義:
Lambda 表達式 也可以成爲閉包,他是推動Java8 發佈的最重要的新特性
Lambda表達式 允許把函數作爲一個方法的參數(函數作爲參數傳遞進方法中),
使用Lambda 表達式可以使代碼變的更加簡潔緊湊.
語法:
(parameters) -> expression
或
(parameters) -> { statements; }
重要特徵
- 可選類型聲明: 不需要聲明參數類型.編譯器可以統一識別參數值
- 可選的參數圓括號: 一個參數無需定義圓括號,但多個參數需要定義圓括號
- 可選的大括號: 如果主體包含了一個語句,就不需要使用大括號
- 可選的返回關鍵字: 如果主體只有一個表達式返回值則編譯器會自動返回值,大括號需要指明表達式返回了一個數值
Lambda 表達式實例
//1.不需要參數,返回值爲 5
() -> 5
//2.接受一個參數(數字類型),返回其2倍的值
x -> 2 * x
//3.接受2個參數(數字),並返回他們的差值
(x,y) -> x - y
//4.接受2個int型整數,返回他們的和
(int x, int y) -> x + y
//5. 接受一個 string 對象,並在控制檯打印,不返回任何值(看起來像是返回void)
(String s) -> System.out.print(s)
package com.wzx.test;
public class Java8Tester {
public static void main(String[] args) {
Java8Tester tester = new Java8Tester();
//類型聲明
MathOperation addition = (int a, int b) -> a + b;
//不用類型聲明
MathOperation subtraction = (a, b) -> a + b;
//大括號中的返回語句
MathOperation multiplication = (int a, int b) -> {
return a + b;
};
//沒有大括號及返回語句
MathOperation division = (int a, int b) -> a + b;
System.out.println("10 + 5 = " + tester.operate(10, 5, addition));
System.out.println("10 + 5 = " + tester.operate(10, 5, subtraction));
System.out.println("10 + 5 = " + tester.operate(10, 5, multiplication));
System.out.println("10 + 5 = " + tester.operate(10, 5, division));
//不用括號
GreetingService greetingService1 = message -> System.out.println("Hello" + message);
//用括號
GreetingService greetingService2 = (message) -> System.out.println("Hello" + message);
greetingService1.sayMessage("Runoob");
greetingService2.sayMessage("Google");
}
interface MathOperation {
int operation(int a, int b);
}
interface GreetingService {
void sayMessage(String message);
}
private int operate(int a, int b, MathOperation mathOperation) {
return mathOperation.operation(a, b);
}
}
//控制檯打印
"C:\Program Files\Java\jdk1.8.0_181\bin\java.exe" "....
10 + 5 = 15
10 + 5 = 15
10 + 5 = 15
10 + 5 = 15
HelloRunoob
HelloGoogle
注意要點:
- Lambda 表達式主要用來定義行內執行的方法類型接口,例如 一個簡單方法接口 在上面例子中 我們使用各種額理性的Lambda 表達式來定義MathOperation 接口的方法,然後我們定義了sayMessage的執行
- Lambda 表達式免去了使用匿名方法的麻煩,並且給予Java簡單但是強大的函數化的編程能力
變量作用域
Lambda 表達式只能引用標記了final的外層局部變量,這就是說不能再Lambda 內部修改定義在域外的局部變量,否則會變異錯誤.
public static void main(String[] args) {
//此處可以不聲明 爲 final 但是之後num 這個變量不能被更改
final int num = 1;
Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));
s.convert(3);
//不可以被更改
// num = 5;
}
public interface Converter<T1, T2> {
void convert(int i);
}
標註:
- Lambda表達式的局部變量可以不用聲明爲final,但是必須不可被後面的代碼修改(即隱性的具有 final 的語義)
- 在Lambda表達式當中不允許聲明一個與局部變量同名的參數或者局部變量
Optional類:
- optional類的引入是主要解決空指針異常(NullPointerException)
- Optional 類是一個可以爲null的容器對象, 如果值存在isPresent() 方法會返回true,調用get()方法會返回該對象
- Optional 是一個容器: 他可以保存類型T的值,或者僅僅保存null. Optional 提供很多有用的方法,這樣我們就不用顯式進行空值檢測
之前寫法:
先定義一個Student類,取年齡大於18 並且 成績大於60的人員姓名
Student student = new Student();
student.setAge(12);
student.setName("wzxzzzzzz");
student.setScore(90);
Student student2 = new Student();
student2.setAge(20);
student2.setName("wzwwwwwww");
student2.setScore(20);
Student student3 = new Student();
student3.setAge(22);
student3.setName("wxsssssss");
student3.setScore(100);
List<Student> listStudent = new ArrayList<>();
listStudent.add(student);
listStudent.add(student2);
listStudent.add(student3);
我們在訪問對象方法或屬性的調用都可能導致NullPointerException 當我們確保不出發異常就需要在訪問之前對其進行空指針檢查:
for (Student studentss : listStudent) {
if (studentss != null) {
if (studentss.getAge() >= 18) {
Integer score = studentss.getScore();
if (score != null && score > 80) {
System.out.println("被選中的名字:" + studentss.getName());
}
}
}
}
Optional寫法 :
for (Student students : listStudent) {
Optional<Student> studentOptional = Optional.of(students);
Integer integer = studentOptional.filter(s -> s.getAge() >= 18).map(Student::getScore).orElse(0);
if (integer > 80) {
System.out.println("被選中的名字:" + students.getName());
}
}
Optional的使用方法:
-
orElse: 存在該值,則返回該值 , 如果值爲空則返回指定的值,
-
orElseGet: 存在該值,則返回該值 , 如果值爲空則調用指定的方法返回
//舉例 Map<String,Object> map = new HashMap<String,Object>(); //如果map不爲空 則返回 map 如果爲空 則調用orElseGet() 方法 new Map Map<String,Object> resultMap = Optional.ofNullable(map).orElseGet(HashMap::new);
-
orElseThrow: 如果值爲空則直接拋出異常
-
Optional ofNullable(T value): 如果爲非空,返回 Optional 描述的指定值,否則返回空的 Optional
-
Optional of(T value): 返回一個指定非null值的Optional
java中遍歷Map的方法:
1: 通過map.keySet()
//先得到key,再通過key去獲得value
Set<Integer> set = map.keySet();
for (Integer integer: set) {
System.out.println(integer +"的對應值爲:"+map.get(integer));
}
2:通過map.values 遍歷所有的value
//通過map.values 遍歷所有的value
for(String value : map.values()){
System.out.println("value值:"+value);
}
3:遍歷map.entrySet
//推薦方法
for(Map.Entry<Integer,String> entry : map.entrySet()){
System.out.println("key:"+entry.getKey()+",value:"+entry.getValue());
}
4:使用Iterator 迭代器和Map entrySet
Iterator<Map.Entry<Integer,String>> iterator = map.entrySet().iterator();
while (iterator.hasNext()) {
Map.Entry<Integer,String> entry = iterator.next();
System.out.println("Iterator方法的key:" + entry.getKey() +",value:" + entry.getValue());
}
5:lambda 表達式
//推薦使用
map.forEach((key, value) -> {
System.out.println(key + ":" + value);
});
java中遍歷List的方法:
public static void main(String[] args) {
ArrayList<Object> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
Iterator iterator = list.iterator();
System.out.println("--------第一種--------");
while (iterator.hasNext()) {
Integer i = (Integer) iterator.next();
System.out.println(i);
}
System.out.println("---------第二種---------");
list.forEach(str -> {
//java8新特性
System.out.println(str);
});
System.out.println("----------第三種--------");
list.parallelStream().forEach(str -> {
//多線程並行執行無序
System.out.println(str);
});
System.out.println("----------第四種--------");
list.stream().forEach(str -> {
//串行執行
System.out.println(str);
});
System.out.println("----------第五種--------");
ArrayList<String> listss = new ArrayList<>();
listss.add("一");
listss.add("二");
listss.add("三");
for (Iterator<String> iterator2 = listss.iterator(); iterator2.hasNext(); ) {
System.out.println(iterator2.next());
}
System.out.println("----------第六種--------");
listss.forEach(new Consumer<String>() {
@Override
public void accept(String str) {
System.out.println(str);
}
});
System.out.println("----------第七種--------");
for (String str : listss) {
System.out.println(str);
}
System.out.println("----------第八種--------");
for (int i = 0; i < listss.size(); i++) {
System.out.println(listss.get(i));
}
}