整体介绍

◆实例讲解Lambda语法 ◆基于Lambda实现函数式编程 ◆Stream流式处理

什么是Lambda表达式

◆JDK8(SDK 1.8)开始支持Lambda表达式,用来让程序编写更优雅 ◆利用Lambda可以更简洁的实现匿名内部类与函数声明与调用 ◆基于Lambda提供streami流式处理极大简化对集合的操作

传统代码

使用Lambda表达式

Lambda语法格式

新建四则运算接口,MathOperation.java

package com.imooc.lambda;
 
/**
 * 四则运算接口
 */
public interface MathOperation {
    public Float operate(Integer a ,Integer b);
}

实现接口,新建LambdaSample.java文件

package com.imooc.lambda;
 
public class LambdaSample {
 
    public static void main(String[] args) {
        //1.标准Lambda使用方式
        //约束条件:Lambda表达式只能实现有且只有一个抽象方法的接口,Java称为"函数式接口"
        MathOperation addition = (Integer a,Integer b)->{
            System.out.println("加法运算");
            return a+b+0f;
        };
        System.out.println(addition.operate(5,3));
        /*等价代码
        1.创建类
        class Addition implements MathOperation{
            @Override
            public Float operate(Integer a, Integer b) {
                System.out.println("加法运算");
                return a+b+0f;
            }
        }
        2.实例化对象
        Addition addition = new Addition();
        3.调用函数
        System.out.println(addition.operate(5,3));
        */
 
        //2.Lambda允许忽略参数类型
        MathOperation substraction = (a,b)->{
            return a-b+0f;
        };
        System.out.println(substraction.operate(5,3));
 
        //3.单行实现代码可以省略大括号和return
        MathOperation multiplication = (a,b)->a*b+0f;
        System.out.println(multiplication.operate(5,3));
 
    }
}

注意:约束条件:Lambda表达式只能实现有且只有一个抽象方法的接口,Java称为”函数式接口” 因为lambda不知道该实现哪个抽象方法

函数式编程

基于Lambda的一种全新开发方式

什么是函数式编程

◆函数式编程是基于函数式接口并使用lambda表达的编程方式 ◆函数式编程理念是将代码作为可重用数据代入到程序运行中 ◆函数式编程强调”你想做什么”,而不是”你想怎么做’

什么是函数式接口

◆函数式接口是有且只有一个抽象方法的接口 ◆Java中拥有大量函数式接口,如java.lang.Runnable ◆JDK8后提供了一系列新的函数式接口,位于java.util.function

函数式接口Predicate

◆Predicate是新增的函数式接口,位于java.util.function ◆Predicate用于测试传入的数据是否满足判断要求 ◆Predicate接口需要实现test()方法进行逻辑判断,返回true或false

package com.imooc.lambda;
 
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
 
/**
 
* 理解函数式编程
* Predicate函数式接口的使用方法
  */
  public class PredicateSample {
  public static void main(String[] args) {
  //内部是泛型 Integer传入T,得知test方法传入需为整形
  Predicate<Integer> predicate = n->n<4;
  boolean result = predicate.test(10);
  System.out.println(result);
  List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
  /*只能判断出奇数类型,局限大
for(Integer num:list){
if(num%2==1){
System.out.println(num);
}
}*/
 
 
  filter(list,n->n%2==1); //取所有奇数
  filter(list,n->n%2==0); //取所有偶数
  filter(list,n->n>5 && n%2==0); //取所有大于5的偶数
 
 
  }
  public static void filter(List<Integer> list , Predicate<Integer> predicate){
  for(Integer num:list){
  if(predicate.test(num)){
  System.out.print(num + " ");
  }
  }
  System.out.println("");
  }
  }

注意Filter函数的第二个参数类型,是predicate接口Predicate<Integer> predicate

其它函数式接口

java.util.function (Java Platform SE 8 ) (oracle.com)

Consumer使用

package com.imooc.lambda;
 
import java.util.function.Consumer;
 
/**
 * Consumer接口的使用
 */
public class ConsumerSample {
    public static void main(String[] args) {
        output(s-> System.out.println("向控制台打印:" + s));
        output(s->{
            System.out.println("向XXX网站发送数据包:" + s);
        });
    }
    public static void output(Consumer<String> consumer){
        String text = "天将降大任于是人也,必先苦其心志,劳其筋骨,饿其体肤,空乏其身,行拂乱其所为。";
        consumer.accept(text);
    }
}

Function<T,R>使用

随机生成32位长度的字符串,常用于安全领域的加密与解密

package com.imooc.lambda;
 
import java.util.Random;
import java.util.function.Function;
 
/**
 * 利用Function函数式接口生成定长随机字符串
 */
public class FunctionSample {
    public static void main(String[] args) {
        Function<Integer,String> randomStringFunction = l->{
          String chars = "abcdefghijklmnopqrstuvxwyz0123456789";
            StringBuffer stringBuffer = new StringBuffer();
            Random random = new Random();
            for(int i = 0 ; i < l ; i++){
                int position = random.nextInt(chars.length());
                stringBuffer.append(chars.charAt(position));
            }
            return stringBuffer.toString();
        };
        String randomString = randomStringFunction.apply(32);
        System.out.println(randomString);
    }
}

总结

Stream介绍

Stream流式处理

◆Stream流式处理是建立在Lambda基础上的多数据处理技术 ◆Stream对集合数据处理进行高度抽象极大简化代码量 ◆Stream可对集合进行迭代,去重,筛选,排序,聚合等一系列处理

Stream实例

流水线式加工代码,第一步筛选偶数,第二步大到小排序,最后找出最大的数

Stream常用方法

Stream的五种创建方式

package com.imooc.stream;
 
import org.junit.Test;
 
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.stream.IntStream;
import java.util.stream.Stream;
 
/**
 * Stream流对象的五种创建方式
 */
public class StreamGenerator {
    //1.基于数组进行创建
    @Test
    public void generator1(){
        String[] arr = {"Lily" , "Andy" , "Jackson" , "Smith"};
        Stream<String> stream = Stream.of(arr);
        stream.forEach(s->System.out.println(s));
    }
 
    //2.基于集合进行创建
    @Test
    public void generator2(){
        List<String> list = new ArrayList<>();
        list.add("Lily");
        list.add("Andy");
        list.add("Jackson");
        list.add("Smith");
        Stream<String> stream = list.stream();
        stream.forEach(s->System.out.println(s));
    }
 
    //3.利用generate方法创建无限长度流
    @Test
    public void generator3(){
        Stream<Integer> stream = Stream.generate(() -> new Random().nextInt(100000));
        stream.limit(10).forEach(i->System.out.println(i));
    }
 
    //4.基于迭代器创建流
    @Test
    public void generator4(){
        Stream<Integer> stream = Stream.iterate(1,n->n+1);
        stream.limit(100).forEach(i->System.out.println(i));
    }
    //5.基于字符序列创建流
    @Test
    public void generator5(){
        String str = "abcdefg我的";
        IntStream stream = str.chars();
        stream.forEach(c -> System.out.println((char)c));
    }
}

stream.of()方法解析生成对应的Stream对象,需要指定泛型类型,此处为String类型 按住ctrl,点击forEach,即可看到内部的实现为Coustom

void forEach(Consumer<? super T> var1);此处s类型由泛型类型可知是String

Stream常用方法

Stream (Java Platform SE 8 ) (oracle.com)

package com.imooc.stream;
 
import org.junit.Test;
 
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
 
public class StreamMethod {
    //提取集合中所有偶数并求和
    @Test
    public void case1(){
        List<String> list = Arrays.asList("1", "2", "3", "4", "5" , "6");
        int sum = list.stream() //获取stream对象
            .mapToInt(s->Integer.parseInt(s)) //mapToInt将流中每一个数据转为整数
            .filter(n->n%2==0) //filter对流数据进行过滤
            .sum();//求和
        System.out.println(sum);
    }
 
    //所有名字首字母大写
    @Test
    public void case2(){
        List<String> list = Arrays.asList("lily","smith","jackson");
        List newList = list.stream()
                //按规则对每一个流数据进行转换
                .map(s->s.substring(0,1).toUpperCase() + s.substring(1))
                //.forEach(s-> System.out.println(s));
                //collect对流数据进行收集,生成新的List/Set
                .collect(Collectors.toList());
        System.out.println(newList);
    }
 
    //将所有奇数从大到小进行排序,且不许出现重复
    @Test
    public void case3(){
        List<Integer> list = Arrays.asList(1,60,38,21,51,60,51,73);
        List newList = list.stream().distinct()//去除重复的流数据
            .filter(n->n%2==1)
            .sorted((a,b)->b-a) //流数据排序
            .collect(Collectors.toList());
        System.out.println(newList);
 
    }
 
}