Java8發佈已經有一段時間了,此次發佈的改動比較大,不少人將此次改動與Java5的升級相提並論。Java8其中一個很重要的新特性就是lambda表達式,容許咱們將行爲傳到函數中。想一想看,在Java8 以前咱們想要將行爲傳入函數,僅有的選擇就是匿名內部類。Java8發佈之後,lambda表達式將大量替代匿名內部類的使用,簡化代碼的同時,更突出了原來匿名內部類中最重要的那部分包含真正邏輯的代碼。尤爲是對於作數據的同窗來講,當習慣使用相似scala之類的函數式編程語言之後,體會將更加深入。如今咱們就來看看Java8中lambda表達式的一些常見寫法。javascript
lambda體中調用方法的參數列表和返回值類型,要和函數式接口中抽象方法的參數列表和返回值類型保持一致。java
1、替代匿名內部類編程
lambda表達式用的最多的場合就是替代匿名內部類,實現Runnable接口是匿名內部類的經典例子。lambda表達式的功能至關強大,用()->就能夠代替整個匿名內部類!c#
package OSChina.Lambda; import org.junit.Test; import java.util.Comparator; import java.util.function.Consumer; import java.util.function.Function; import java.util.function.Predicate; import java.util.function.Supplier; public class Test1{ public static void main(String[] args) { Runnable runnable = new Runnable() { @Override public void run() { System.out.println("普通,線程啓動"); } }; runnable.run(); test2(); test3(); test4(); test5(); } //無參數,無返回值 public static void test2() { //「->」左邊只有一個小括號,表示無參數,右邊是Lambda體(就至關於實現了匿名內部類裏面的方法了,(即就是一個可用的接口實現類了。)) Runnable runnable = ()->System.out.println("Lambda 表達式方式,線程啓動"); runnable.run(); } //有一個參數,而且無返回值 public static void test3() { //這個e就表明所實現的接口的方法的參數, Consumer<String> consumer = e->System.out.println("Lambda 表達式方式,"+e); consumer.accept("傳入參數"); } //有兩個以上的參數,有返回值,而且 Lambda 體中有多條語句 public static void test4() { //Lambda 體中有多條語句,記得要用大括號括起來 Comparator<Integer> com = (x, y) -> { System.out.println("函數式接口"); return Integer.compare(x, y); }; int compare = com.compare(100, 244); System.out.println("有兩個以上的參數,有返回值,"+compare); } //若 Lambda 體中只有一條語句, return 和 大括號均可以省略不寫 public static void test5() { //Comparator com = (x, y) -> Integer.compare(100, 244); System.out.println("若 Lambda 體中只有一條語句, return 和 大括號均可以省略不寫,"+Integer.compare(100, 244)); } }
2、Java8四大內置函數式接口數組
若是使用lambda還要本身寫一個接口的話太麻煩,因此Java本身提供了一些接口:app
一、Consumer 消費性接口:void accept(T t);dom
//有一個參數,而且無返回值 public static void test3() { //這個e就表明所實現的接口的方法的參數, Consumer<String> consumer = e->System.out.println("Lambda 表達式方式,"+e); consumer.accept("傳入參數"); }
二、Supplier供給型接口: T get();編程語言
package OSChina.Lambda; import java.util.ArrayList; import java.util.function.Supplier; public class Test2 { public static void main(String[] args) { ArrayList<Integer> res = getNumList(10,()->(int)(Math.random()*100)); System.out.println(res); } public static ArrayList<Integer> getNumList(int num, Supplier<Integer> sup){ ArrayList<Integer> list = new ArrayList<>(); for (int i = 0; i < num; i++) { Integer e = sup.get(); list.add(e); } return list; } }
三、Function 函數式接口:R apply(T t);ide
package OSChina.Lambda; import java.util.function.Function; public class Test2 { public static void main(String[] args) { String newStr = strHandler("abc",(str)->str.toUpperCase()); System.out.println(newStr); newStr = strHandler(" abc ",(str)->str.trim()); System.out.println(newStr); } public static String strHandler(String str, Function<String,String>fun){ return fun.apply(str); } }
四、Predicate 斷言式接口:boolean test(T t);函數式編程
判斷一些字符串數組判斷長度>2
的字符串:
package OSChina.Lambda; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.function.Predicate; public class Test2 { public static void main(String[] args) { List<String> list = Arrays.asList("hello","jiangshuying","lambda","www","ok","q"); List<String> ret = filterStr(list,(str)->str.length()>2); System.out.println(ret); } public static List<String> filterStr(List<String> list, Predicate<String> pre){ ArrayList<String> arrayList = new ArrayList<>(); for(String str:list){ if(pre.test(str)) { arrayList.add(str); } } return arrayList; } }
3、方法引用與構造器引用
要求:實現抽象方法的參數列表和返回值類型,必須與方法引用的方法的參數列表和返回值類型保持一致!
方法引用:使用操做符「::」將類與方法分隔開來。
對象::實例方法名
類::靜態方法名
類::實例方法名
舉個例子:
public static void test9(){ Comparator<Integer> comparator = (x,y)->Integer.compare(x,y); Comparator<Integer> comparator1 = Integer::compare; int compare = comparator.compare(1,2); int compare1 = comparator1.compare(1,2); System.out.println("compare:"+compare); System.out.println("compare1:"+compare1); }
4、lambda表達式的一些常見用法
一、使用lambda表達式對集合進行迭代
package OSChina.Lambda; import java.util.Arrays; import java.util.List; public class Test3 { public static void main(String[] args) { List<String> list = Arrays.asList("java","c#","javascript"); //before java8 for (String str:list){ System.out.println("before java8,"+str); } //after java8 list.forEach(x-> System.out.println("after java8,"+x)); } }
二、用lambda表達式實現map
map函數能夠說是函數式編程裏最重要的一個方法了。map的做用是將一個對象變換爲另一個。在咱們的例子中,就是經過map方法將cost增長了0,05倍的大小而後輸出。
package OSChina.Lambda; import java.util.Arrays; import java.util.List; public class Test3 { public static void main(String[] args) { List<Double> list = Arrays.asList(10.0,20.0,30.0); list.stream().map(x->x+x*0.05).forEach(x-> System.out.println(x)); } }
三、用lambda表達式實現map與reduce
既然提到了map,又怎能不提到reduce。reduce與map同樣,也是函數式編程裏最重要的幾個方法之一。。。map的做用是將一個對象變爲另一個,而reduce實現的則是將全部值合併爲一個,請看:
package OSChina.Lambda; import java.util.Arrays; import java.util.List; public class Test3 { public static void main(String[] args) { //before java8 List<Double> cost = Arrays.asList(10.0, 20.0,30.0); double sum = 0; for(double each:cost) { each += each * 0.05; sum += each; } System.out.println("before java8,"+sum); //after java8 List<Double> list = Arrays.asList(10.0,20.0,30.0); double sum2 = list.stream().map(x->x+x*0.05).reduce((sum1,x)->sum1+x).get(); System.out.println("after java8,"+sum2); } }
相信用map+reduce+lambda表達式的寫法高出不止一個level。
四、filter操做
filter也是咱們常用的一個操做。在操做集合的時候,常常須要從原始的集合中過濾掉一部分元素。
package OSChina.Lambda; import java.util.Arrays; import java.util.List; import java.util.stream.Collectors; public class Test3 { public static void main(String[] args) { List<Double> cost = Arrays.asList(10.0, 20.0,30.0,40.0); List<Double> filteredCost = cost.stream().filter(x -> x > 25.0).collect(Collectors.toList()); filteredCost.forEach(x -> System.out.println(x)); } }
五、與函數式接口Predicate配合
除了在語言層面支持函數式編程風格,Java 8也添加了一個包,叫作 java.util.function。它包含了不少類,用來支持Java的函數式編程。其中一個即是Predicate,使用 java.util.function.Predicate 函數式接口以及lambda表達式,能夠向API方法添加邏輯,用更少的代碼支持更多的動態行爲。Predicate接口很是適用於作過濾。
package OSChina.Lambda; import java.lang.reflect.Array; import java.util.Arrays; import java.util.List; import java.util.function.Predicate; public class Test4 { public static void filterTest(List<String> languages, Predicate<String> condition) { languages.stream().filter(x -> condition.test(x)).forEach(x -> System.out.println(x + " ")); } public static void main(String[] args) { List<String> languages = Arrays.asList("Java","Python","scala","Shell","R"); filterTest(languages,x->x.startsWith("J"));//Java filterTest(languages,x -> x.endsWith("a"));//Java,scala filterTest(languages,x -> true);//Java,Python,scala,Shell,R filterTest(languages,x -> false);// filterTest(languages,x -> x.length() > 4);//Python,scala,Shell, } }