lamda表达式与函数式接口

lamda表达式

lamda表达式,也叫做闭包,lamda表达式允许把函数做为一个方法的参数
语法:

(parameters) -> expression
或
(parameters) ->{ statements; }
  • 可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。
  • 可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号,没有参数也必须定义。
  • 可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
  • 可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定表达式返回了一个数值。
  • 当参数、statements只有一个时,可以省略括号
  • 当函数需要返回值,而statements只有一句时,可以省略return关键字。
    表达式实例如下:
//最简单的,什么都不干的函数
()->{};
//可以省略括号
x -> x++;
x -> x++
//可以省略return,
x->{ return x++;} equals  x-> x++;
//不可省略,当statements不只含有1句时,不可省略
x->{x++; return x++}
//无参与多参写法
()-> x++;
 x-> x++;
(x,y)->x+y;

上面介绍了lamda表达式的结构和写法,那么我们如何把lamda表达式当作参数传递给方法呢,接着看

	public void test(){
    Inter1 m=(a,b)->a+b; //定义一个变量接收lamda表达式
   	invokeLamda(1,1,m);
    }
    interface Inter1 {
        int operation(int a, int b);
    }
    public void invokeLamda(a,b,Inter1 inter){ 
    	inter.operation(a,b);
    }
//输出:5

//Java1.8之前,需要灵活的执行不同的方法,只能采取匿名内部类的方式
public void test(){
    son  m=new Son();
    System.out.println(m.operation(3,2));
    }
    interface Inter1 {
        int operation(int a, int b);
    }
    class  son implements Inter1{
    @Override
        public int operation(int a,int b){
            return a+b;
        }
    }

可以看到,lamda表示式,为我们节约了不少的代码量

什么是函数式接口

指的是有且只有一个未实现的方法的接口,一般通过FunctionalInterface这个注解来表明某个接口是一个函数式接口,当然,不用注解标注也是可以的,不过一般还是推荐加上, 假如接口中存在多个抽象函数时,编译器会报错提醒,函数式接口是Java支持函数式编程的基础

Consumer c = (o) -> {
    System.out.println(o);
};
  • 输入:->前面的部分,即被()包围的部分。此处只有一个输入参数,实际上输入是可以有多个的,如两个参数时写法:(a, b);当然也可以没有输入,此时直接就可以是()。
  • 函数体:->后面的部分,即被{}包围的部分;可以是一段代码。
  • 输出:函数式编程可以没有返回值,也可以有返回值。如果有返回值时,需要代码段的最后一句通过return的方式返回对应的值。
    当函数体中只有一个语句时,可以去掉{}进一步简化
Consumer c = (o) -> System.out.println(o);

如果是调用静态方法,可以省略入参

Consumer c = System.out::println;

函数式接口种类

在1.8之前我们就经常使用的Runnable等函数接口,JDK1.8新增了更多函数式接口,如rt.jar包下的java.util.function包里面包含的类,consumer、supplier等,下面简单介绍一下

Consumer

Consumer类预定义了一个抽象方法,接收一个参数,并无返回值,使用方式如下

Consumer c=System.out::println
//控制参数类型
Consumer<String> d=System.out::println;
//双参数函数接口
BiConsumer<Integer,Integer> c=(x,y)->System.out.print(x+y);
//实际经常使用的方式
    public void test(){
        invokeLamda(System.out::println);
    }
    public void invokeLamda(Consumer c){
        int a=3;
        c.accept(a);
    }
    //output:3

Consumer接口还提供了一个andthen()具有函数体的默认方法(jdk1.8出现),在执行完accept函数后,再执行andthen 的函数,做一些额外的操作,源码如下

    default Consumer<T> andThen(Consumer<? super T> after) {
        Objects.requireNonNull(after);
        return (T t) -> { accept(t); after.accept(t); };
    }

使用如下

    public void test(){
         invokeLamda(System.out::println,System.out::println);
    }
    public void invokeLamda(Consumer c,Consumer d){
        int a=3;
        c.andThen(d).accept(a);
    }
    //output:3 
    //		 3

所以使用使用函数接口还是比较简单的,只需要大概了解一下种类,再选择相应的类传入就好了,如果预定义的种类不符合业务需求,可以自己定义函数式接口,加上@FunctionalInterface,需要注意的是,这个注解不是必要的,加了后,编译器只是会检查定义的接口是否符合规范

Function

接收一个参数,并提供返回值,使用如下:
    public void test(){
        invokeLamda(x->x*x);
    }
        public void invokeLamda(Function<Integer,Integer> c){
        int a=3;
        a=c.apply(a);
        System.out.print(a);
    }

如果我们的入参需要在咱们传的函数执行前,先处理一下,比如3+3 参数变为6,写法如下

    public void test(){
        invokeLamda(x->x*x,x->x+x);
    }
        public void invokeLamda(Function<Integer,Integer> c,Function<Integer,Integer> d){
        int a=3;
        a=c.compose(d).apply(a);
        System.out.print(a);
    }
    //输出   36

可以看到,先是执行compose,后是执行了apply,前后依次调用了,同样Function里也有andThen方法,写法一样

a=c.andThen(d).apply(a);

但执行顺序是在apply执行后完成

supplier

无参,并返回一个参数

   public void test(){
        invokeLamda(()->3);
    }
        public void invokeLamda(Supplier<Integer> c){
        int a= c.get();
        System.out.print(a);
    }

Predicate

接收一个参数,返回boolean,使用如下
   public void test(){
        invokeLamda((a)->a>5);
    }
   public void invokeLamda(Predicate<Integer> c){
        int a=5;
        boolean returnflag= c.test(a);
        System.out.print(returnflag);
    }

每一个predicate相当于一个表达式,多个表达式的链接,使用如下

    public void test(){
        invokeLamda((a)->a>5,(a)->false,(a)->true);
    }
    public void invokeLamda(Predicate<Integer> b,Predicate<Integer> ...c){
        int a=5;
        for(Predicate<Integer> temp:c){
           b=b.or(temp);
        }
        boolean returnflag=b.test(a);
        System.out.print(returnflag);
    }
    //true

上面的判断,在1.8以前的写法中为

returnflag= a>5||false||true;

Predicate不仅可以或运算,也可以执行与,非,等于,几种条件的逻辑运算,不一一说明了

其他种类

DoubleBinaryOperator
代表了作用于两个double值操作符的操作,并且返回了一个double值的结果。

DoubleConsumer
代表一个接受double值参数的操作,并且不返回结果。

DoubleFunction<R>
代表接受一个double值参数的方法,并且返回结果

DoublePredicate
代表一个拥有double值参数的boolean值方法

DoubleSupplier
代表一个double值结构的提供方

这里不过多赘述了,网上有详细的结构说明,也可以自行查看java.util.function包下查看源码

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值