Java Review (十六、面向对象----Lambda 表达式)

简介: Java Review (十六、面向对象----Lambda 表达式)

   

文章目录

Lambda 允许把函数作为一个方法的参数(函数作为参数传递到方法中)。

Lambda表达式入门

用匿名内部类实现命令模式

 Command接口

public interface Command {
   //接口里定义的方法相当于处理封装行为
  int progress(int[] target);
}

ProgressArray.java

public class ProgressArray {
  public ProgressArray() {    
  } 
  public void progress(int[] target,Command  cmd) {
    cmd.progress(target);
  }
}

CommandInnerTest.java

public class CommandInnerTest {
  public CommandInnerTest() {
  }
  public static void main(String[] args) {
  ProgressArray progressArray=new ProgressArray();
  int[] array= new int[] {3, -4, 6, 4} ;
  //处理数组,具体处理行为取决于匿名内部类
  progressArray.progress(array, new Command() {
    @Override
    public void progress(int[] target) {
      int sum=0;
      int[] a=target;
      for(int j=0;j<a.length;j++ ) {
         int temp=a[j];
         sum+=temp;
      }
           System.out.println("数组元素的总和:"+sum);
    }
  });
}
}

Lambda表示式实现命令模式

Lambda 表达式可用于简化创建匿名内部类对象:

CommandLambdaTest.java

public class CommandLambdaTest {
  public static void main(String[] args) {
    ProgressArray progressArray = new ProgressArray();
    int[] array = { 3, -4, 6, 4 };
    // 处理数组,具体行为取决于匿名内部类
    progressArray.progress(array, (int[] target) -> {
      int sum = 0;
      for (int temp : array) {
        sum += temp;
      }
      System.out.println("数组元素总和为" + sum);
    });
  }
}

当使用 Lambda 表达式代替匿名内部类创建对象时, Lambda 表达式的代码

块将会代替实现抽象方法的方法体, Lambda 表达式就相当一个匿名方法。

从上面语法格式可以看出, Lambda 表达式的主要作用就是代替匿名内部类的烦琐语法。它分三部分组成。

  • 形参列表 。形参列表允许省略形参类型 。如果形参列表中只有一个参数,甚至连形参列表的圆括号也可以省略。
  • 箭头 —>。必须通过英文中画线和大于符号组成。
  • 代码块。如果代码块只包含一条语句, Lambda 表达式允许省略代码块的花括号,那么这条语句就不要用花括号表示语句结束。 Lambda 码块只有一条 return语句,甚至可以省略return关键字, Lambda 表达式会自动返回这条语句的值。

Lambda语法简单实例

public class LambdaTest {
  public static void main(String[] args) {
    LambdaTest tester=new LambdaTest();
    // 类型声明
    MathOperation addition = (int a, int b) -> a + b;
    // 不用类型声明
    MathOperation subtraction = (a, b) -> a - b;
    // 大括号中的返回语句
    MathOperation multiplication = (int a, int b) -> {
      return a * b;
    };
    // 没有大括号及返回语句
    MathOperation division = (int a, int b) -> a / b;
    System.out.println("10 + 5 = " + tester.operate(10, 5, addition));
    System.out.println("10 - 5 = " + tester.operate(10, 5, subtraction));
    System.out.println("10 x 5 = " + tester.operate(10, 5, multiplication));
    System.out.println("10 / 5 = " + tester.operate(10, 5, division));
    // 不用括号
    GreetingService greetService1 = message -> System.out.println("Hello " + message);
    // 用括号
    GreetingService greetService2 = (message) -> System.out.println("Hello " + message);
    greetService1.sayMessage("Runoob");
    greetService2.sayMessage("Google");
  }
  interface MathOperation {
    int operation(int a, int b);
  }
  interface GreetingService {
    void sayMessage(String message);
  }
  private int operate(int a, int b, MathOperation mathOperation) {
    return mathOperation.operation(a, b);
  }
}

Lambda 表达式与函数式接口

Lambda 表达式的类型,也被称为"目标类型( target type) ", Lambda 表达式的目标类型必须是"函数式接口(  functional interface ) "。 函数式接口代表只包含一个抽象方法的接口 。  函数式接口可以包含多个默认方法、类方法,但只能声明一个抽象方法 。

如果采用匿名内部类语法来创建函数式接口的实例,则只需要实现一个抽象方法,在这种情况下即可采用 Lambda  表达式来创建对象,该表达式创建出来的对象的目标类型就是这个函数式接口 。 查询 Java 8的 API 文档,可以发现大量的函数式接口,例如:  Runnable、 ActionListener 等接口都是函数式接口 。

Java 8专门为函数式接口提供了 @FunctionalInterface 注解,该注解通常放在接口定义前面,该注解对程序功能没有任何作用,它用于告诉编译器执行更严格检查一一检查该接口必须是函数式接口 ,否则编译器就会报错。

由于 Lambda 表达式的结果就是被当成对象 , 因此程序中完全可以使用 Lambda 表达式进行赋值,

例如如下代码 :

// Runnable 接口中只包含一个无参数的方法
// Lambda 表达式代表的匿名方法实现了 Runnable 接口中唯一的、无参数的方法
// 因此下面的 Lambda 表达式创建了 一个 Runnable 对象
Runnable r = () -> {
   for(int i = 0 ; i < 100 ; i ++){
   System.out.println() ;
 }
} 

Lambda 表达式实现的是匿名方法,因此它只能实现特定函数式接口中的唯一方法 。 这意味着 Lambda 表达式有如下两个限制 :

  • Lambda 表达式的目标类型必须是明确的函数式接口 。
  • Lambda 表达式只能为函数式接口创建对象 。 Lambda 表达式只能实现一个方法 , 因此它只能为只有一个抽象方法的接口(函数式接口)创建对象 。

为了保证 Lambda 表达式的目标类型是一个明确的函数式接口,可以有如下三种常见方式 。

  • 将 Lambda 表达式赋值给函数式接口类型的变量 。
  • 将 Lambda 表达式作为函数式接口类型的参数传给某个方法 。
  • 使用函数式接口对 Lambda 表达式进行强制类型转换。
Object obj1 = (Runnable) () - > {
  for(int i = 0 ; i < 100 ; i ++{
   System.out.println(i);
  }   

方法引用与构造器引用

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

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

方法引用和构造器引用可以让 Lambda 表达式的代码块更加简洁 。 方法引用和构造器引用都需要使用两个英文冒号。

Lambda 表达式支持的方法引用和构造器引用

image.png

引用类方法

//函数式接口
@FunctionalInterface
interface Converter{
//将 String 参数转换为Integer
  Integer convert(String from);
}
// 下面代码使用 Larnbda 表达式创建 Converter 对象
Converter converterl = from -> Integer.valueOf(from);
//调用 converterl 对象的 convertO方法将字符串转换为整数
Integer val = converterl.convert( " 99 " );
System.out.println (val); // 输出整数 99

可以如下替换:

//方法引用代替 Larnbda 表达式:引用类方法
//函数式接口中被实现方法的全部参数传给该类方法作为参数
Converter converterl = Integer::valueOf;

引用特定对象的实例方法

// 下面代码使用 Larnbda 表达式创建 Converter 对象
Converter converter2 = from - > " fkit.org ".indexOf (from) ;
//调用 converterl 对象的 convertO方法将字符串转换为整数
Integer value = converter2.convert( "it "};
System.out.println(value} ;   //输出 2

可以如下替换:

// 方法引用代替 Lambda 表达式 : 引用特定对象 的实例方法
// 函数式接口中被实现方法的全部参数传给该方法作为参数
Converter converter2 = "fkit.org": :indexOf ;

引用某类对象的实例方法

//定义函数式接口
@Functionallnterface
interface MyTest{
  //根据 String 、 int 、 int 三个参数生成一个 String返回值
  String test(String a , int b , int c);
}  
// 下面代码使用 Lambda 表达式创建 MyTest 对象
MyTest mt = (a , b , c) - > a.substring(b , c};
//调用mt的test()方法
String str = mt.test( "Java 1 Love you" , 2 , 9) ;
System.out.println(str} ; // 输出 : va 1 Lo

可以如下替换:

// 方法引用代替 Lambda 表达式: 引 用某类对象 的实例方法
// 函数式接口中被实现方法的第一个参数作为调用者
// 后面的参数全部传给该方法作为参数
MyTest mt = String ::substring;

引用构造器

@Funct ionallnterface
interface YourTest{
//根据 String 参数生成一个 JFrame 返回值
 JFrame win(String title);
// 下面代码使用 Lambda 表达式创建 YourTest 对象
YourTest yt = (String a) - > new JFrame(a);
JFrame jf = yt.win( "我的窗口 ") ;
System.out.println(jf) ;

可以如下替换:

// 构造器引用代替 Lambda 表达式
//函数式接口中被实现方法的全部参数传给该构造器作为参数
YourTest yt = JFrame ::new;

Lambda 表达式与匿名内部类的联系和区别

从前面介绍可以看出, Lambda 表达式是匿名内部类的一种简化 , 因此它可以部分取代匿名内部类的作用。

Lambda 表达式与匿名内部类存在如下相同点 :

  • Lambda 表达式与匿名内部类一样,都可以直接访问 “effectively final” 的局部变量,以及外部类的成员变量(包括实例变量和类变量〉 。
  • Lambda 表达式创建的对象与匿名内部类生成的对象一样,都可以直接调用从接口中继承的默认方法 。

Lambda 表达式与匿名内部类主要存在如下区别:

  • 匿名内部类可以为任意接口创建实例一一不管接口包含多少个抽象方法,只要匿名内部类实现所有的抽象方法即可;但 Lambda 表达式只能为函数式接口创建实例。
  • 匿名内部类可以为抽象类甚至普通类创建实例 ; 但 Lambda 表达式只能为函数式接口创建实例 。
  • 匿名内部类实现的抽象方法的方法体允许调用接口中定义的默认方法 ; 但 Lambda 表达式的代码块不允许调用接口中定义的默认方法。


参考:

【1】:《疯狂Java讲义》

【2】:https://www.jb51.net/article/171183.htm

【3】:https://www.runoob.com/java/java8-lambda-expressions.html

【4】:《Java核心技术 卷一》


目录
打赏
0
0
0
0
21
分享
相关文章
|
1月前
|
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
32 1
Java中的Lambda表达式与Stream API的协同作用
在本文中,我们将探讨Java 8引入的Lambda表达式和Stream API如何改变我们处理集合和数组的方式。Lambda表达式提供了一种简洁的方法来表达代码块,而Stream API则允许我们对数据流进行高级操作,如过滤、映射和归约。通过结合使用这两种技术,我们可以以声明式的方式编写更简洁、更易于理解和维护的代码。本文将介绍Lambda表达式和Stream API的基本概念,并通过示例展示它们在实际项目中的应用。
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
深入理解Java中的Lambda表达式
在Java 8中引入的Lambda表达式,不仅简化了代码编写,还提升了代码可读性。本文将带你探索Lambda表达式背后的逻辑与原理,通过实例展示如何高效利用这一特性优化你的程序。
探索Java中的Lambda表达式
本文将深入探讨Java 8引入的Lambda表达式,这一特性极大地简化了代码编写,提高了程序的可读性。通过实例分析,我们将了解Lambda表达式的基本概念、使用场景以及如何优雅地重构传统代码。文章不仅适合初学者,也能帮助有经验的开发者加深对Lambda表达式的理解。
Java中的Lambda表达式:简化代码的现代魔法
在Java 8的发布中,Lambda表达式的引入无疑是一场编程范式的革命。它不仅让代码变得更加简洁,还使得函数式编程在Java中成为可能。本文将深入探讨Lambda表达式如何改变我们编写和维护Java代码的方式,以及它是如何提升我们编码效率的。
|
2月前
|
Java中的Lambda表达式与Stream API的高效结合####
探索Java编程中Lambda表达式与Stream API如何携手并进,提升数据处理效率,实现代码简洁性与功能性的双重飞跃。 ####
37 0
探索Java中的Lambda表达式与Stream API
【10月更文挑战第22天】 在Java编程中,Lambda表达式和Stream API是两个强大的功能,它们极大地简化了代码的编写和提高了开发效率。本文将深入探讨这两个概念的基本用法、优势以及在实际项目中的应用案例,帮助读者更好地理解和运用这些现代Java特性。
Java 8新特性之Lambda表达式与Stream API的探索
【9月更文挑战第24天】本文将深入浅出地介绍Java 8中的重要新特性——Lambda表达式和Stream API,通过实例解析其语法、用法及背后的设计哲学。我们将一探究竟,看看这些新特性如何让Java代码变得更加简洁、易读且富有表现力,同时提升程序的性能和开发效率。
|
5月前
|
Java 8新特性:Lambda表达式与Stream API的深度解析
【7月更文挑战第61天】本文将深入探讨Java 8中的两个重要特性:Lambda表达式和Stream API。我们将首先介绍Lambda表达式的基本概念和语法,然后详细解析Stream API的使用和优势。最后,我们将通过实例代码演示如何结合使用Lambda表达式和Stream API,以提高Java编程的效率和可读性。