龙空技术网

Lambda表达式详解

千锋IT小助手 78

前言:

此刻小伙伴们对“lambda表达式教程”大致比较讲究,各位老铁们都想要剖析一些“lambda表达式教程”的相关文章。那么小编在网上网罗了一些关于“lambda表达式教程””的相关资讯,希望咱们能喜欢,小伙伴们一起来了解一下吧!

1 Lambda表达式是Java8中的新特性

Java8中引入Lambda表达式,使得java可以函数式编程,在并发性能上迈出了实质性的一步。

什么是函数式编程?函数式编程(英语:functional programming)或称函数程序设计,又称泛函编程,是一种编程范型,它将电脑运算视为数学上的函数计算,并且避免使用程序状态以及易变对象。函数编程语言最重要的基础是λ演算(lambda calculus)。而且λ演算的函数可以接受函数当作输入(引数)和输出(传出值)。

ps:λ这个符号可以在搜狗输入法的符号中显示

而在面向对象编程中,面向对象程序设计(英语:Object-oriented programming,缩写:OOP)是种具有对象概念的程序编程范型,同时也是一种程序开发的方法。它可能包含数据、属性、代码与方法。对象则指的是类的实例。它将对象作为程序的基本单元,将程序和数据封装其中,以提高软件的重用性、灵活性和扩展性,对象里的程序可以访问及经常修改对象相关连的数据。在面向对象程序编程里,计算机程序会被设计成彼此相关的对象。

2 lambda表达式能干什么?

  @FunctionalInterface  public interface InterfaceA {       void show();  }  public class Demo {      public static void main(String[] args) {          InterfaceA a1 = new InterfaceA() {  ​              @Override              public void show() {                  System.out.println("匿名内部类的实现方式");  ​              }          };          a1.show();          InterfaceA a2 = () -> {              System.out.println("我是lambda表达式");          };  ​          a2.show();      }  ​  }  ​

使用匿名内部类的一个问题是:当一个匿名内部类的实现非常简单,比如说接口只有一个抽象函数 ,那么匿名内部类的语法有点笨拙且不清晰。

使用下面这种方法的时候,不需要再使用new XXX(){}这种繁琐代码,不需要指出重写的方法的名字,也不需要给出重写方法的返回值类型,只要给出重写的方法括号以及括号里的形参列表即可

从上面介绍考科一看出,当使用Lambda表达式代替匿名内部类创建对象时,Lambda表达式的代码块将会代替实现抽象方法的方法体,Lambda表达式就相当于一个匿名方法

3 语法

(参数)->表达式 或 (参数)->{方法体;}

1.形参列表:

形参列表允许省略形参类型,若形参列表中只有一个参数,形参列表的圆括号也可以省略代码

2.箭头(->)

必须通过英文中划线号和大于符号组成

3.代码块:

如果代码块只包含一条语句,lambda表达式允许省略代码块的花括号,那么这条语句就不要用花括号表示语句结束

lambda代码块只有一条return语句,甚至可以省略return关键字

lambda表达式需要返回值,而它的代码块中仅有一条省略了return的语句,lambda表达式会自动返回这条语句的结果

lambda表达式的写法:

  interface InterfaceA{      void showA();  }  interface InterfaceB{      void showB(int ia);  }  interface InterfaceC{       void showC(int ib,int ic);  }  ​  public class Demo {  ​      public static void main(String[] args) {         InterfaceA a1 = ()->{             System.out.println("接口A中方法");         };         //简化:         InterfaceA a2 = ()-> System.out.println("接口A中方法");         a1.showA();         a2.showA();                  InterfaceB b1 = (int ia)->{             System.out.println("接口B中方法:"+ia);         };         //简化:         InterfaceB b2 = (ia)->{             System.out.println("接口B中方法:"+ia);         };         InterfaceB b3 = (ia)-> System.out.println("接口B中方法:"+ia);                  InterfaceB b4 = ia -> System.out.println("接口B中方法:"+ia);         b1.showB(10);         b2.showB(20);         b3.showB(30);         b4.showB(40);                           InterfaceC c1 = (int ia,int ib)->{             System.out.println("接口B中方法:"+ia+"和"+ib);         };         //简化         InterfaceC c2 = (ia,ib)->{             System.out.println("接口B中方法:"+ia+"和"+ib);         };         InterfaceC c3 = (ia,ib)-> System.out.println("接口B中方法:"+ia+"和"+ib);         c1.showC(1,2);         c2.showC(3,4);         c3.showC(5,6);      }  ​  }

lambda表达式就可以把函数当做函数的参数,代码(函数)当做数据(形参),这种特性满足上述需求。当要实现只有一个抽象函数的接口时,使用lambda表达式能够更灵活。

  interface InterfaceA{      void showA();  }  interface InterfaceB{      void showB(int ia);  }  interface InterfaceC{       void showC(String ib,int ic);  }  ​  public class Demo {      public static void showInterfaceA(InterfaceA a) {            System.out.println(a);            a.showA();      }      public static void showInterfaceB(String message,InterfaceB b) {          System.out.println("b信息:"+b+message);          b.showB(100);      }      public static void showInterfaceC(String message,int phoneNumber,InterfaceC c) {          System.out.println("c信息:"+c+message);          c.showC(message, phoneNumber);      }  ​      public static void main(String[] args) {         showInterfaceA(()->System.out.println("可以作为参数传递"));         showInterfaceB("接口B的lambda",b->System.out.println("可以作为参数传递"+b));         showInterfaceC("接口C的lambda",88888888,(message,phoneNumber)->System.out.println("可以作为参数传递"+message+phoneNumber));      }  ​  }
4 Lambda表达式与函数式接口

在上面的案例中.方法的参数的数据类型或是获取一个对象,但是在实际调用中我们传入的是一个lambda表达式,可以发现程序可以

正常编译,运行,这说明Lambda表达式实际上将会被当成一个"类型"的对象

Lambda表达式的类型,也被称为"目标类型(target type)",Lambda表达式的目标类型必须是"函数式接口(functional interface)"

ps:Java8新引入的概念,函数接口(functional interface)。它的定义是:一个接口,如果只有一个显式声明的抽象方法,那么它就是一个函数接口。一般用@FunctionalInterface标注出来 (也可以不标记),函数式接口可以包含多个default或static方法,但是只能声明一个抽象方法

@FuctionalInterface主要作用就是检查当前接口是不是函数接口

若想使用lambdaname目标必须是一个函数接口

5 Lambda表达式引用全局和局部变量

  @FunctionalInterface  interface VariableTest {      void sayMessage(String message);  }  ​  public class Demo {      //静态全局变量      static String ms1 = "Hello!";      //实例全局变量      String ms2 = "Hello";      public static void main(String[] args) {          VariableTest vt = message -> {              //直接在lambda表达式中调用全局变量              System.out.println(message+ms1);              System.out.println(message+new Demo().ms2);                        };          vt.sayMessage("Lambda ");          //局部变量-->在lambda表达式中抵用的不是局部变量而是常量          String ms3 = "hello";          VariableTest vt2 = message -> {              //直接在lambda表达式中调用常量              System.out.println(message+ms3);              //ms3 = "llo";//修改报错 因为是常量,若是要修改这个 会提示显示声明添加 final          };          vt2.sayMessage("Lambda ");                          }  ​  }

6 方法引用与构造器引用

如果Lambda表达式的代码块只有一条代码,程序就可以省略Lambda表达式中的代码块的花括号

不仅如此,如果Lambda表达式的代码块只有一条代码,还可以在代码块中使用方法引用和构造器引用

方法引用和构造器引用都需要使用::两个英文冒号

6.1 引用类方法

在函数式接口中定义的抽象方法,而方法的实现是触发某个类.方法(调用类方法的形式)来完成时可以使用

  @FunctionalInterface  interface Converter{      //将字符串转化对应的数      Integer convert(String str);  }  ​  public class Demo {      public static void main(String[] args) {         Converter c1 = str -> Integer.valueOf(str);         Integer val1 = c1.convert("69");         System.out.println(val1);         //类方法应用来代替Lambda表达式,函数式接口中被实现方法的全部参数传递给改类方法作为参数         Converter c2 = Integer::valueOf;         Integer val2 = c2.convert("57");         System.out.println(val2);      }  }  ​
6.2 引用特定对象的实例方法

在函数式接口中定义的抽象方法,而方法的实现是触发对象.方法(调用类方法的形式)来完成时可以使用

  @FunctionalInterface  interface ClassATest {      void dispaly(String message);  }  class A{      public void show(String message) {          System.out.println("调用了特定对象的实例方法"+message);      }  }  public class Demo {            public static void main(String[] args) {         ClassATest a1 = message -> new A().show(message);         a1.dispaly("21");         //特定对象的实例方法引用代替Lambda表达式,函数式接口中实现方法的全部参数传给该方法作为参数         ClassATest a2 = new A()::show;         a2.dispaly("17");      }  ​  }

6.3 引用某类对象的实例方法

在函数式接口中定义的抽象方法,而方法的实现是触发是方法中第一个参数的对象.方法(调用类方法的形式)来完成时可以使用

  @FunctionalInterface  interface SubString {      String show(String str,int b,int c);  }  ​  public class Demo {            public static void main(String[] args) {          SubString su1 = (str,b,c) -> str.substring(b, c);          String val = su1.show("I Love U", 2, 6);          System.out.println(val);          //某类对象的实例方法引用代替Lambda表达式,函数式接口中被实现方法的第一个参数作为调用者,后面的参数全部传递给改方法          SubString su2 = String::substring;          String val2 = su2.show("I Love U", 2, 6);          System.out.println(val2);      }  ​  }  ​
6.4 引用构造方法

在函数式接口中定义的抽象方法,而方法的返回值是一个对应类的实例

  @FunctionalInterface  interface ClassBTest {      B getInstance(String name,int age);  }  class B{      private String name;      private int age;      public B(String name, int age) {          this.name = name;          this.age = age;      }      @Override      public String toString() {          return "我是创建好的对象年龄"+age+"姓名"+name;      }  }  public class Demo {            public static void main(String[] args) {          ClassBTest cb1 = (name,age)->new B(name,age);          B b1 = cb1.getInstance("张三", 18);          System.out.println(b1);          //构造方法引用代替lambda表达式,函数式接口中被实现方法的全部参数传该构造方法作为参数          ClassBTest cb2 = B::new;          B b2 = cb2.getInstance("李四", 19);          System.out.println(b2);                }  ​  }
7 Lambda表达式和匿名内部类的区别

1.匿名内部类可以为任意接口创建实例,不管接口包含多少个抽象方法,只要匿名内部类实现所有的抽象方法即可

但Lambda表达式只能为函数式接口创建实例(即只能有一个抽象方法)

2.匿名内部类可以为抽象类甚至是普通类创建实例

但Lambda表达式只能为函数式接口创建实例

3.匿名内部类实现的抽象方法的方法体允许调用接口中定义的默认(default)方法

但Lambda表达式的代码块不允许调用接口中的默认(default)方法

8 Lambda表达式应用

  String[] name = {"张三", "李四",  "王五", "赵六"};    List<String> players =  Arrays.asList(name);        // 以前的循环方式    for (String player : players) {         System.out.print(player + "; ");    }        // 使用 lambda 表达式以及函数操作  Java8中新方法  players.forEach((player) -> System.out.print(player + "; "));         // 在 Java 8 中使用双冒号操作符    players.forEach(System.out::println  ​  ​  // 1.1使用匿名内部类    new Thread(new Runnable() {        @Override        public void run() {            System.out.println("Hello world !");        }    }).start();        // 1.2使用 lambda expression    new Thread(() -> System.out.println("Hello world !")).start();        // 2.1使用匿名内部类    Runnable race1 = new Runnable() {        @Override        public void run() {            System.out.println("Hello world !");        }    };        // 2.2使用 lambda expression    Runnable race2 = () -> System.out.println("Hello world !");         // 直接调用 run 方法(没开新线程哦!)    race1.run();    race2.run();  ​  String[] names = {"张三", "李四",  "王五", "赵六"};        // 1.1 使用匿名内部类根据 name 排序 names    Arrays.sort(names, new Comparator<String>() {        @Override        public int compare(String s1, String s2) {            return (s1.compareTo(s2));        }    });  // 1.2 使用 lambda 排序 names    Comparator<String> sortByName = (String s1, String s2) -> (s1.compareTo(s2));    Arrays.sort(names, sortByName);        // 1.3 也可以采用如下形式:    Arrays.sort(names, (String s1, String s2) -> (s1.compareTo(s2)));   

标签: #lambda表达式教程