Java8函数式编程之Lambda表达式

简介: 笔记

一、Lambda表达式简介


函数式编程是种编程范式,它将计算机运算视为函数的计算。函数编程语言最重要的基础是λ演算(lambda calculus)。而且λ演 算的函数可以接受函数当作输入(参数)和输出(返回值)。和指令式编程相比,函数式编程强调函数的计算比指令的执行重 要。和过程化编程相比,函数式编程里,函数的计算可随时调用。


(1)示例

为了理解Lambda表达式的概念,下面先从一个示例开始。

假设有这样的一个需求:设计一个通用方法,能够实现两个数值的加法和减法运算。Java中方法不能 单独存在,必须定义在类或接口中,考虑是一个通用方法,可以设计一个数值计算接口,其中定义该 通用方法,代码如下:

//可计算接口
public interface Calculable {
//    计算两个int值
    int calculateInt(int a, int b);
    /**
     * 通过操作符进行计算
     * @param opr 操作符
     * @return 实现 Calculable接口对象
     */
    public static Calculable calculable(char opr){
        Calculable result;
        if (opr == '+'){
//            匿名内部类实现Calculable接口
            result = new Calculable() {
//                实现加法运算
                @Override
                public int calculateInt(int a, int b) {
                    return a+b;
                }
            };
        }else{
//            内部类实现Calculable接口
            result = new Calculable() {
                @Override
                public int calculateInt(int a, int b) {
                    return a-b;
                }
            };
        }
        return result;
    }
}

调用通用方法代码如下:

//测试代码
public class HelloWorld {
    public static void main(String[] args) {
        int n1 = 10;
        int n2 = 5;
//        实现加法计算Calculable对象
        Calculable c1 = Calculable.calculable('+');
        //        实现减法计算Calculable对象
        Calculable c2 = Calculable.calculable('-');
//        调用calculateInt方法进行计算
        System.out.printf("%d + %d=%d \n",n1,n2, c1.calculateInt(n1,n2));
        //        调用calculateInt方法进行计算
        System.out.printf("%d - %d=%d \n",n1,n2,c2.calculateInt(n1,n2));
    }
}


(2)Lambda表达式实现

通过匿名内部类实现通用方法calculate代码很臃肿,Java 8采用Lambda表达式可以替代匿名内

部类。修改之后的通用方法calculate代码如下:

//可计算接口
@FunctionalInterface
public interface Calculable {
//    计算两个int值
    int calculateInt(int a, int b);
    /**
     * 通过操作符进行计算
     * @param opr 操作符
     * @return 实现 Calculable接口对象
     */
    public static Calculable calculable(char opr){
        Calculable result;
        if (opr == '+'){
//            Lambda表达式实现Calculable接口
            result = (int a, int b) ->{
                return a+b;
            };
        }else{
//            Lambda表达式实现Calculable接口
            result = (int a, int b) ->{
                return a-b;
            };
        }
        return result;
    }
}


二、Lambda表达式简化形式


使用Lambda表达式是为了简化程序代码,Lambda表达式本身也提供了多种简化形式,这些简化形式虽然简化了代码,但客观上使得代码可读性变差。

(1)省略参数类型

//可计算接口
@FunctionalInterface
public interface Calculable {
//    计算两个int值
    int calculateInt(int a, int b);
    /**
     * 通过操作符进行计算
     * @param opr 操作符
     * @return 实现 Calculable接口对象
     */
    public static Calculable calculable(char opr){
        Calculable result;
        if (opr == '+'){
//            Lambda表达式实现Calculable接口
            result = ( a,  b) ->{
                return a+b;
            };
        }else{
//            Lambda表达式实现Calculable接口
            result = ( a, b) ->{
                return a-b;
            };
        }
        return result;
    }
}


(2)省略参数小括号

Lambda表达式中参数只有一个时,可以省略参数小括号。


(3)省略return和大括号

如果Lambda表达式体中只有一条语句,那么可以省略return和大括号,代码如下:

public static Calculable calculate(int power) {
    Calculable result;
  if (power == 2) {
// Lambda表达式实现Calculable接口 
    result = (int a) -> { //标准形式
            return a * a;
        };
  } else {
// Lambda表达式实现Calculable接口
    result = a -> a * a * a; //省略形式 1
  };
    return result;
}


三、作为参数使用Lambda表达式


Lambda表达式一种常见的用途是作为参数传递给方法。这需要声明参数的类型声明为函数式接口类型。

代码如下:

public class HelloWorld1 {
    public static void main(String[] args) {
        int n1 = 10;
        int n2 = 5;
        // 打印计算结果加法计算结果
        diaplay((a,b)->{
            return a+b;
        },n1,n2);
        // 打印计算结果减法计算结果
        diaplay((a,b)->a-b,n1,n2);
    }
/**
 * 打印计算结果 *
 * @param calc Lambda表达式
 * @param n1 操作数1
 * @param n2 操作数2 */
    public static void diaplay(Calculable calc,int n1,int n2){
        System.out.println(calc.calculateInt(n1,n2));
    }
}

参数calc类型是Calculable,这个参数即可以接收 实现Calculable接口的对象,也可以接收Lambda表达式,因为Calculable是函数式接口。


四、访问变量


Lambda表达式可以访问所在外层作用域内定义的变量,包括:成员变量和局部变量。


(1)访问成员变量

成员变量包括:实例成员变量和静态成员变量。在Lambda表达式中可以访问这些成员变量,此时的Lambda表达式与普通方法一样,可以读取成员变量,也可以修改成员变量。

public class LambdaDemo {
    // 实例成员变量
    private int value = 10;
    // 静态成员变量
    private static int staticValue = 5;
    // 静态方法,进行加法运算
    public static Calculable add() {
        Calculable result = (int a, int b) -> {
            // 访问静态成员变量,不能访问实例成员变量
            staticValue++;
            int c = a + b +staticValue;
            return c;
        };
        return result;
    }
    // 实例方法,进行减法运算
    public Calculable sub() {
        Calculable result = (int a, int b) -> {
            // 访问静态成员变量和实例成员变量
            staticValue++;
            this.value++;
            int c = a - b - staticValue - this.value;
            return c; 
        };
        return result;
    }
}


(2)捕获局部变量

对于成员变量的访问Lambda表达式与普通方法没有区别,但是对于访问外层局部变量时,会发生“捕 获变量”情况。Lambda表达式中捕获变量时,会将变量当成final的,在Lambda表达式中不能修改那些捕获的变量。 示例代码如下:

public class LambdaDemo {
    // 实例成员变量
    private int value = 10;
    // 静态成员变量
    private static int staticValue = 5;
    // 静态方法,进行加法运算
    public static Calculable add() {
    //局部变量
    int localValue = 20;
        Calculable result = (int a, int b) -> {
            int c = a + b +staticValue;
            return c;
        };
        return result;
    }
    // 实例方法,进行减法运算
    public Calculable sub() {
      //final局部变量
    final int localValue = 20;
        Calculable result = (int a, int b) -> {
            // 访问静态成员变量和实例成员变量
      staticValue++;
      this.value++;
            int c = a - b - staticValue - this.value;
            return c;
        };
        return result;
    }
}


五、方法引用


Java 8之后增加了双冒号“::”运算符,该运算符用于“方法引用”,注意不是调用方法。“方法引用”虽然没有直接使用Lambda表达式,但也与Lambda表达式有关,与函数式接口有关。 方法引用分为:静态方法的方法引用和实例方法的方法引用。它们的语法形式如下:

类型名::静态方法 // 静态方法的方法引用 
实例名::实例方法 // 实例方法的方法引用


被引用方法的参数列表和返回值类型,必须与函数式接口方法参数列表和方法返回值类型 一致。

代码如下:

public class LambdaDemo {
    // 静态方法,进行加法运算
// 参数列表要与函数式接口方法calculateInt(int a, int b)兼容
    public static int add(int a, int b){
        return a + b;
    }
    // 实例方法,进行减法运算
// 参数列表要与函数式接口方法calculateInt(int a, int b)兼容
    public int sub(int a, int b) {
        return a - b;
    }
}

LambdaDemo类中提供了一个静态方法add,一个实例方法sub。这两个方法必须与函数式接口方法参数 列表一致,方法返回值类型也要保持一致。

代码如下:

public class HelloWorld {
    public static void main(String[] args) {
        int n1 = 10;
        int n2 = 5;
        // 打印计算结果加法计算结果
        display(LambdaDemo::add, n1, n2);
        LambdaDemo d = new LambdaDemo();
        // 打印计算结果减法计算结果
        display(d::sub, n1, n2);
    }
    /**
     * 打印计算结果
     * @param calc Lambda表达式 * @param n1 操作数1
     * @param n2 操作数2
     */
    public static void display(Calculable calc, int n1, int n2) {
        System.out.println(calc.calculateInt(n1, n2));
    }
}



相关文章
|
4月前
|
Java 大数据 API
Java Stream API:现代集合处理与函数式编程
Java Stream API:现代集合处理与函数式编程
313 100
|
5月前
|
安全 Java API
Java中的Lambda表达式:简洁与功能的结合
Java中的Lambda表达式:简洁与功能的结合
487 211
|
5月前
|
安全 Java
Java中的Switch表达式:更简洁的多路分支
Java中的Switch表达式:更简洁的多路分支
536 211
|
5月前
|
Java 编译器
Java 17 Switch表达式:更简洁、更强大的流程控制
Java 17 Switch表达式:更简洁、更强大的流程控制
726 111
|
4月前
|
Java 开发者
Java 函数式编程全解析:静态方法引用、实例方法引用、特定类型方法引用与构造器引用实战教程
本文介绍Java 8函数式编程中的四种方法引用:静态、实例、特定类型及构造器引用,通过简洁示例演示其用法,帮助开发者提升代码可读性与简洁性。
|
5月前
|
Java 编译器 API
Java Lambda表达式与函数式编程入门
Lambda表达式是Java 8引入的重要特性,简化了函数式编程的实现方式。它通过简洁的语法替代传统的匿名内部类,使代码更清晰、易读。本文深入讲解Lambda表达式的基本语法、函数式接口、方法引用等核心概念,并结合集合操作、线程处理、事件回调等实战案例,帮助开发者掌握现代Java编程技巧。同时,还解析了面试中高频出现的相关问题,助你深入理解其原理与应用场景。
|
6月前
|
自然语言处理 Java Apache
在Java中将String字符串转换为算术表达式并计算
具体的实现逻辑需要填写在 `Tokenizer`和 `ExpressionParser`类中,这里只提供了大概的框架。在实际实现时 `Tokenizer`应该提供分词逻辑,把输入的字符串转换成Token序列。而 `ExpressionParser`应当通过递归下降的方式依次解析
392 14
|
6月前
|
设计模式 数据采集 Java
Java正则表达式的基础知识,进阶至熟练掌握。
通过大量的练习来熟悉它们的识别模式、如何设计模式来解决实际问题,才能够逐步达到熟练掌握。更多的是通过实践、编写代码和解决真实问题来完善技能。在这方面,没有快速的捷径,唯有刻意练习和长时间的代码实践。
123 0
|
7月前
|
SQL JSON 安全
Java 8 + 中 Lambda 表达式与 Stream API 的应用解析
摘要:本文介绍了Java 8+核心新特性,包括Lambda表达式与Stream API的集合操作(如过滤统计)、函数式接口的自定义实现、Optional类的空值安全处理、接口默认方法与静态方法的扩展能力,以及Java 9模块化系统的组件管理。每个特性均配有典型应用场景和代码示例,如使用Stream统计字符串长度、Optional处理Map取值、模块化项目的依赖声明等,帮助开发者掌握现代Java的高效编程范式。(150字)
151 1