Lambda总结

转载请注明地址:http://www.jianshu.com/p/86dc38f0cec8

第一部分、方法引用

方法引用总体上分为四种:

  • 类之中静态方法的引用 类名称 :: 静态方法名称
  • 类之中普通方法的引用 实例化对象名称::普通方法
  • 类中构造方法的引用 类名称 ::new
  • 特定类型的任意方法的引用 类名称::方法名称

1、简化的lambda——方法引用(Method Reference)

lambda已经简化了代码的写法,然而方法引用进一步简化了lambda的写法。
方法引用的使用方式:类名::方法名

类型 使用方式 备注
引用静态方法 ContainingClass::staticMethodName Integer::valueOf简化了i->Integer.valueOf(i)的写法
引用特定对象的实例方法 containingObject::instanceMethodName s::toString()简化了()->s.toString()
引用特定类型的任意对象的实例方法 ContainingType::methodName System.out::println简化了(s)->System.out.println(s),其中System.out表示的是PrintStream对象
引用构造函数 ClassName::new String::new简化了()->new String()

例1:

@FunctionalInterface
interface myfun{
    public P fun(T t);
}
class Test {
    public static void main(String[] args) {
        //类之中的静态方法
        myfun test1 = String::valueOf; //valueOf
        //等价于 myfun test1 =s->String.valueOf(s);
        System.out.println(test1.fun(1000));    
        //类之中普通方法
        String string = "Hello World .";
        myfun test2 = string::startsWith;
    }
}

当然这里的接口myfun,在jdk8中为我们提供了Function的接口,我们可以直接使用。

例:

Class Demo{
   public static  Integer getInteger(String s) {
        Function function = (a) -> Integer.parseInt(a);
        return function.apply(s);

    }
}

更简化的写法如下:

Class Demo{
   public static  Integer getInteger(String s) {
        Function function = Integer::parseInt(a);
        return function.apply(s);

    }
}

第二部分、内建式函数式接口

  • java提供的核心内建函数式接口
  • 集合的新型输出操作

核心接口

  1. 函数型接口 Function
  2. 消费型接口 Consumer
  3. 供给型接口 Supplier
  4. 断言型接口 Predicate

举例:

    List list = new ArrayList();
        list.add("A");
        list.add("B");
        list.add("C");
        list.forEach(s->System.out.println(s));

第三部分、数据流

1、构造流的几种常见方法
// 1. Individual values
Stream stream = Stream.of("a", "b", "c");

// 2. Arrays
String [] strArray = new String[] {"a", "b", "c"};
stream = Stream.of(strArray);
stream = Arrays.stream(strArray);

// 3. Collections
List list = Arrays.asList(strArray);
stream = list.stream();

构造数据流,总体上可分为三种方法。

  • 直接利用Stream的of方法,将几个独立的值转化为数据流。

  • 利用数组

  • 利用Collection接口,由于Collection接口有个stream方法。

    default Stream stream();
    

    stream()方法定义如下:

     default Stream stream() {
            return StreamSupport.stream(spliterator(), false);
        }
    
2、数值流的构造
IntStream.of(new int[]{1, 2, 3}).forEach(System.out::println);
IntStream.range(1, 3).forEach(System.out::println);
IntStream.rangeClosed(1, 3).forEach(System.out::println);
3、流转换为其它数据结构
// 1. Array
String[] strArray1 = stream.toArray(String[]::new);
// 2. Collection
List list1 = stream.collect(Collectors.toList());
List list2 = stream.collect(Collectors.toCollection(ArrayList::new));
Set set1 = stream.collect(Collectors.toSet());
Stack stack1 = stream.collect(Collectors.toCollection(Stack::new));
// 3. String
String str = stream.collect(Collectors.joining()).toString();
Stream> inputStream = Stream.of(
 Arrays.asList(1),
 Arrays.asList(2, 3),
 Arrays.asList(4, 5, 6)
 );
Stream outputStream = inputStream.
flatMap((childList) -> childList.stream());
reduce 的用例
// 字符串连接,concat = "ABCD"
String concat = Stream.of("A", "B", "C", "D").reduce("", String::concat); 
// 求最小值,minValue = -3.0
double minValue = Stream.of(-1.5, 1.0, -3.0, -2.0).reduce(Double.MAX_VALUE, Double::min); 
// 求和,sumValue = 10, 有起始值
int sumValue = Stream.of(1, 2, 3, 4).reduce(0, Integer::sum);
// 求和,sumValue = 10, 无起始值
sumValue = Stream.of(1, 2, 3, 4).reduce(Integer::sum).get();
// 过滤,字符串连接,concat = "ace"
concat = Stream.of("a", "B", "c", "D", "e", "F").
 filter(x -> x.compareTo("Z") > 0).
 reduce("", String::concat);

1、使用Stream静态方法来创建Stream

1.1、 of方法:有两个overload方法,一个接受变长参数,一个接口单一值

//1
Stream integerStream = Stream.of(1, 2, 3, 5);
Stream integerStream = Stream.of("A", "B", "C");
//2
Stream stringStream = Stream.of("taobao");

1.2、generator方法:生成一个无限长度的Stream,其元素的生成是通过给定的Supplier(这个接口可以看成一个对象的工厂,每次调用返回一个给定类型的对象)

Stream.generate(newSupplier() {
    @Override
    publicDouble get() {
    returnMath.random();
    }
});

Stream.generate(() -> Math.random());

Stream.generate(Math::random);

三条语句的作用都是一样的,只是使用了lambda表达式和方法引用的语法来简化代码。每条语句其实都是生成一个无限长度的Stream,其中值是随机的。这个无限长度Stream是懒加载,一般这种无限长度的Stream都会配合Stream的limit()方法来用

1.3、iterate方法:也是生成无限长度的Stream,和generator不同的是,其元素的生成是重复对给定的种子值(seed)调用用户指定函数来生成的。其中包含的元素可以认为是:seed,f(seed),f(f(seed))无限循环

Stream.iterate(1, item -> item + 1).limit(10).forEach(System.out::print);
//结果如下12345678910

2、通过Collection子类获取Stream

这个在本文的第一个例子中就展示了从List对象获取其对应的Stream对象,如果查看Java doc就可以发现Collection接口有一个stream方法,所以其所有子类都都可以获取对应的Stream对象。

publicinterfaceCollection extendsIterable {
    //其他方法省略
    defaultStream stream() {
        returnStreamSupport.stream(spliterator(), false);
    }
}

流的几种构造方法:

// 1. Individual values
Stream stream = Stream.of("a", "b", "c");
// 2. Arrays
String [] strArray = new String[] {"a", "b", "c"};
stream = Stream.of(strArray);
stream = Arrays.stream(strArray);
// 3. Collections
List list = Arrays.asList(strArray);
stream = list.stream();

例:

Arrays.asList("tony", "cafei", "aaron")//将字符串数组转换为列表
              .stream()
              .map(str -> str.toUpperCase())
              .forEach(it -> System.out.println(it));

也可以简化成如下写法

   Arrays.asList("tony", "cafei", "aaron")//将字符串数组转换为列表
           .stream()
           .map(String::toUpperCase)
           .forEach(System.out::println);

参考:Stream语法详解

你可能感兴趣的:(Lambda总结)