简单使用lambda表达式

目录

什么是lambda表达式?

 下面先来个简单案例:

简化lambda表达式:

总结:

语法格式一 :无参,无返回值

语法格式二: :Lambda 需要一个参数,但是没有返回值

语法格式三 : 数据类型可以省略

语法格式四: :Lambda 若只需要一个参数时, 参数的小括号可以省略

语法格式五: :Lambda 需要两个或以上的参数,多条执行语句,并且可以有返回值

语法格式六 :当 当 Lambda 体只有 一条, 语句时,return 号 与大括号 若有,都可以省略

然后下面说下函数式接口:

简单使用lambda表达式使用四大函数式接口:

 方法引用与构造器引用


什么是lambda表达式?

Lambda 是一个 匿名函数,我们可以把 Lambda 表达式理解为是 一段可以传递的代码(将代码像数据一样进行传递)。

使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升

Lambda 表达式:在Java 8 语言中引入的一种新的语法元素和操作符。这个操作符为 “->” , 该操作符被称为 Lambda 操作符或 箭 头操作符。

它将 Lambda 分为两个部分:左侧:指定了 Lambda 表达式需要的参 参 数 列表: 右侧:指定了 Lambda 体 ,是抽象方法的实现逻辑,也即Lambda 表达式要执行的功能

 下面先来个简单案例:

package com.chen.lambda;


/*
* 推导lambda表达式
* */
public class TestLambda {

    //3.静态内部类
   static class Like2 implements ILike{
        @Override
        public void lambda() {
            System.out.println("I like lambda2");
        }
    }

    public static void main(String[] args) {
        ILike like=new Like();
        like.lambda();

        like=new Like2();
        like.lambda();


        //4.局部内部类
        class Like3 implements ILike{
            @Override
            public void lambda() {
                System.out.println("I like lambda3");
            }
        }

        like=new Like3();
        like.lambda();


        //5.匿名内部类,没有类的名称,必须帮助接口或者父类
        like =new ILike() {
            @Override
            public void lambda() {
                System.out.println("I like lambda4");
            }
        };
        like.lambda();


        //6.用lambda简化
        like=()->{
            System.out.println("I like lambda5");
        };
        like.lambda();

    }



}

//1.定义一个函数式接口
interface ILike{

    void lambda();

}
//2.实现类
class Like implements ILike{
    @Override
    public void lambda() {
        System.out.println("I like lambda");
    }
}

简化lambda表达式:

package com.chen.lambda;

public class TestLambda2 {

    static class Love implements ILove{

        @Override
        public void love(int a) {
            System.out.println("I Love you -->"+a);
        }
    }

    public static void main(String[] args) {
        ILove love=new Love();
        love.love(2);

        //匿名内部类
        ILove love2=new ILove() {
            @Override
            public void love(int a) {
                System.out.println("I Love you -->"+a);
            }
        };
        love2.love(5);


        //lambda表达式简化
        ILove love3=(int a)->{
            System.out.println("I Love you -->"+a);
        };
        love3.love(520);

        //简化1.参数类型
        ILove love4=(a)->{
            System.out.println("I Love you -->"+a);
        };
        love4.love(520);

        //简化2.简化括号
        ILove love5=a ->{
            System.out.println("I Love you -->" + a);
        };
        love5.love(520);

        //简化3.去掉花括号
        ILove love6=a ->
            System.out.println("I Love you -->" + a);
        love6.love(520);

        //总结:
          //lambda表达式只能有一行代码的情况下才能称为一行,如果有多行,那么就用代码块包裹
          //前提是接口为函数式接口
          //多个参数也可以去掉参数类型,要去掉就都去掉,必须加上括号

        ILove2 love7=(a,b) ->
               a*b;
      System.out.println(love7.love(7,8));
    }

}


interface ILove{
    void love(int a);

}
interface ILove2{
   int love(int a,int b);

}

class Love implements ILove{

    @Override
    public void love(int a) {
        System.out.println("I Love You");
    }
}

总结:

语法格式一 :无参,无返回值

  Runnable runnable1=()->System.out.println("ddd");

语法格式二: :Lambda 需要一个参数,但是没有返回值

Consumer<String> con=(String str) -> {System.out.println(str);};

//输出
   con.accept("aaa");

语法格式三 : 数据类型可以省略

​Consumer<String> con=(str) -> {System.out.println(str);};

语法格式四: :Lambda 若只需要一个参数时, 参数的小括号可以省略

Consumer<String> con=str -> {System.out.println(str);};

语法格式五: :Lambda 需要两个或以上的参数,多条执行语句,并且可以有返回值

 Comparator<Integer> com=(x, y) -> {
            System.out.println("实现函数式接口方法");
            return Integer.compare(x,y);
        };
  //输出结果
      System.out.println(com.compare(11,10));

语法格式六 :当 当 Lambda 体只有 一条, 语句时,return 号 与大括号 若有,都可以省略

 Comparator<Integer> com=(x, y) -> Integer.compare(x,y);
     

上述 Lambda 表达式中的参数类型都是由编译器推断得出的。Lambda
表达式中无需指定类型,程序依然可以编译,这是因为 javac 根据程序
的上下文,在后台推断出了参数的类型。Lambda 表达式的类型依赖于
上下文环境,是由编译器推断出来的。这就是所谓的“类型推断”

然后下面说下函数式接口:

只包含一个抽象方法的接口,称为 函数式接口。
1.你可以通过 Lambda 表达式来创建该接口的对象。(若 Lambda 表达式抛出一个受检异常(即:非运行时异常),那么该异常需要在目标接口的抽象方法上进行声明)。
2.我们可以在一个接口上使用 @FunctionalInterface 注解,这样做可以检查它是否是一个函数式接口。同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。

在Java8中,有所不同。在
Java8中,Lambda表达式是对象,而不是函数,它们必须依附于一类特别的对象类型——函数式接口。
简单的说,在Java8中,Lambda表达式就是一个函数式接口的实例。这就是Lambda表达式和函数式接口的关系。

也就是说,只要一个对象是函数式接口的实例,那么该对象就可以用Lambda表达式来表示

 所以以前用匿名实现类表示的现在都可以用Lambda表达式来写

 

简单使用lambda表达式使用四大函数式接口:

代码案例:

//Consumer<T> 消费型接口:
    @Test
    public void test1(){
        happy(10000,(m) -> System.out.println("今天消费了:"+(m+m)));
    }

    public  void happy(double money, Consumer<Double> con){

        con.accept(money);
    }



    //Supplier 供给型接口:
    @Test
    public void test2(){
        List<Integer> numList=  getNumList(10,() ->(int)(Math.random()*100));
        for(Integer num:numList){
            System.out.println(num);
        }
    }
    //需求:产生指定个整数,并放入到集合中
    public List<Integer> getNumList(int num, Supplier<Integer> sup){
        List<Integer>  list=new ArrayList<>();

        for(int i=0;i<num;i++){
            Integer n=sup.get();
            list.add(n);
        }
        return list;
    }



    //Function<T ,R> 函数型接口
    @Test
    public void test3(){
     String newStr  =   strHandler("\t\t\t 我爱爱学习",(str) -> str.trim());
     System.out.println(newStr);

        String subStr = strHandler("\t\t\t 我爱爱学习",(str)-> str.substring(2,5));
        System.out.println(subStr);
    }

    //需求:用于处理字符串
    public String strHandler(String str, Function<String,String> fun){

        return fun.apply(str);

    }

   //Predicate<T> 断言型接口

    @Test
    public  void test4(){
        List<java.lang.String> list= Arrays.asList("hello","hello2","hello3");
         List<String> strList=filterStr(list,(s) -> s.length()>5);

         for(String str:strList){
             System.out.println(str);
         }
    }

    //需求:将满足条件的字符串,放入集合中
    public List<String> filterStr(List<String> list, Predicate<String> pre){

        List<String> strList=new ArrayList<>();

        for(String str:list){
            if(pre.test(str)){
                strList.add(str);
            }
        }
      return  strList;

    }

 方法引用与构造器引用

方法引用

当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用!
1.方法引用可以看做是Lambda表达式深层次的表达。换句话说,方法引用就是Lambda表达式,也就是函数式接口的一个实例,通过方法的名字来指向
一个方法,可以认为是Lambda表达式的一个语法糖。
2.要求:实现接口的抽象方法的参数列表和返回值类型,必须与方法引用的方法的参数列表和返回值类型保持一致!

3.格式:使用操作符 “::” 将类(或对象) 与 方法名分隔开来。
 如下三种主要使用情况:
 对象:: 实例方法名
 类 :: 静态方法名
 类 :: 实例方法

例如:

Consumer<String> con=(x) -> System.out.println(x);

//相当于

Consumer<String> con=System.out::println;
Comparator<Integer> c=(x,y) -> Integer.compare(x,y);

//相当于

Comparator<Integer> c2=Integer::compare;
BiPredicate<String,String> bp=(x,y) -> x.equals(y);

//相当于

BiPredicate<String,String> bp1=String::equals;
//输出结果
 System.out.println(bp1.test("s","t"));

构造器 引用

格式: ClassName::new
与函数式接口相结合,自动与函数式接口中方法兼容。
可以把构造器引用赋值给定义的方法,要求构造器参数列表要与接口中抽象
方法的参数列表一致!且方法的返回值即为构造器对应类的对象

Function<Integer,Myclass> fun=(n) -> new Myclass(n);

//相当于

Function<Integer,Myclass> fun=Myclass::new;

数组引用

Function<Integer,Integer[]> fun3=(n) -> new Integer[n];

//相当于

Function<Integer,Integer[]> fun4=Integer[]::new;
   System.out.println(fun4.apply(1555).length);

猜你喜欢

转载自blog.csdn.net/qq_44716544/article/details/119047362