【Java 干货教程】Java方法引用详解

导言

Java方法引用是Java 8引入的一项重要特性,它提供了一种简洁、可读性高的方式来直接引用已经存在的方法。方法引用使得代码更加简洁、易懂,同时提高了代码的可维护性和重用性。本文将详细介绍Java方法引用的概念、语法和使用方法,并提供一些示例代码。

一、方法引用的概念

方法引用是一种直接引用已经存在的方法的方式,它允许我们在代码中通过方法的名称来引用方法。方法引用可以被看作是Lambda表达式的一种简化形式,它提供了一种更加简洁的方式来实现函数式接口。

在Java中,方法引用主要用于简化函数式接口的实现,特别是当我们需要将一个方法作为参数传递给另一个方法时,使用方法引用可以使代码更加清晰。

二、方法引用的语法

方法引用的语法由两部分组成:类名或对象名和方法名,中间使用双冒号(::)进行分隔。根据方法引用的情况,可以分为以下几种形式:

  • 静态方法引用:类名::静态方法名
  • 实例方法引用:对象名::实例方法名
  • 特定类的任意对象方法引用:类名::实例方法名
  • 构造方法引用:类名::new

 2.1、静态方法引用

使用类名和双冒号(::)来引用一个静态方法。例如,Math::max表示引用Math类中的静态max方法。

Weigher<Integer, Integer> integerIntegerWeigher = Math::max;

2.2、实例方法引用

使用实例对象和双冒号(::)来引用一个非静态的实例方法。例如,String::length表示引用String对象的length方法。

LineHandler lineHandler = String::length;

2.3、特定类的任意对象方法引用

使用特定对象的实例和双冒号(::)来引用该对象的实例方法。例如,myObject::methodName表示引用myObject对象的methodName方法。

Student student = new Student();
Copier<String> getUsername = student::getUsername;

2.4、构造方法引用

Copier<ArrayList> arrayListCopier = ArrayList::new;

三、方法引用的使用方法

方法引用主要用于函数式接口,即只包含一个抽象方法的接口。函数式接口可以使用方法引用来实现抽象方法,从而实现相应的功能。

Java中提供了许多函数式接口,如ConsumerPredicateSupplier等。可以使用方法引用来简化这些接口的实现,使代码更加简洁、易读。

下面是一些方法引用的使用示例:

3.1、使用方法引用实现Consumer接口

List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
numbers.forEach(System.out::println);

forEach()方法源码

default void forEach(Consumer<? super T> action) {
        Objects.requireNonNull(action);
        for (T t : this) {
            action.accept(t);
        }
    }

这里的Consumer就是函数式接口

3.2、使用方法引用实现Supplier接口

Supplier<List<String>> listSupplier = ArrayList::new;
List<String> names = listSupplier.get();

Supplier也是函数式接口(源码)

@FunctionalInterface
public interface Supplier<T> {

    /**
     * Gets a result.
     *
     * @return a result
     */
    T get();
}

四、案例 

以下是一个示例Java程序,演示了Java方法引用的四种使用形式:

import java.util.Arrays;
import java.util.List;
import java.util.function.Supplier;

class Vehicle {
    private String type;

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }

    // 静态方法
    public static void printName(String name) {
        System.out.println(name);
    }

    // 实例方法
    public void printPowerValue(Long powerValue) {
        System.out.println(powerValue);
    }

    @Override
    public String toString() {
        return "Vehicle{" +
                "type='" + type + '\'' +
                '}';
    }
}

public class MethodReferenceExample {

    public static void main(String[] args) {
        // 静态方法引用: 类名::静态方法
        List<String> names = Arrays.asList("擎天柱", "霸天虎", "威震天");
        names.forEach(Vehicle::printName);
        System.out.println();

        // 实例方法引用: 对象名::实例方法
        Vehicle example = new Vehicle();
        List<Long> messages = Arrays.asList(50000L, 120000L);
        messages.forEach(example::printPowerValue);
        System.out.println();

        // 构造方法引用: 类名::new
        Supplier<Vehicle> vehicleSupplier = Vehicle::new;
        // 创建对象
        Vehicle car = vehicleSupplier.get();
        car.setType("Car");
        Vehicle bike = vehicleSupplier.get();
        bike.setType("Bike");
        // 打印对象
        System.out.println(car);
        System.out.println(bike);
        System.out.println();

        // 引用特定类型的任意对象的实例方法: 对象名::实例方法
        String prefix = "Prefix:";
        List<String> suffixes = Arrays.asList("A", "B", "C");
        suffixes.stream()
                .map(prefix::concat) // 使用String的concat方法
                .forEach(System.out::println); // 使用System.out的println方法
    }
}

在这个示例中,我们使用了静态方法引用、实例方法引用、构造方法引用和引用特定类型的任意对象的实例方法。通过运行这个程序,可以看到各个方法引用形式的使用效果。

运行结果:

擎天柱
霸天虎
威震天

50000
120000

Vehicle{type='Car'}
Vehicle{type='Bike'}

Prefix:A
Prefix:B
Prefix:C

五、应用场景

方法引用广泛应用于函数式编程、Stream API和Lambda表达式等方面,特别在以下几个场景中被普遍使用:

1. 集合操作:在集合的排序、筛选、映射等操作时,可以使用方法引用简化代码;
2. GUI编程:在事件处理、回调函数等方面,通过方法引用来实现事件的响应机制;
3. 数据处理:在对数据进行加工、处理和计算时,方法引用常用于替代复杂的Lambda表达式。

想了解Stream API 和Lambda表达式和方法引用的场景可以看看我的这篇文章

【Java干货教程】Stream流详解-CSDN博客

六、注意事项

在使用方法引用时,有一些注意事项需要考虑

  1. 方法引用只能用于函数式接口:方法引用只能用于那些只有一个抽象方法的接口,也就是函数接口。确保在使用方法引用时,目标类型是一个函数接口,否则编译将会失败。
  2. 参数匹配:方法引用的参数必须与目标方法的参数列表兼容。这意味着参数数量和类型要一致。如果目标方法具有多个参数,可以通过上下文推断或者函数式接口的泛型来进行类型匹配。(方法引用的方法签名必须与函数式接口的抽象方法的参数列表和返回类型一致
  3. 方法引用可能引发空指针异常:在使用实例方法引用时,请确保对象引用不是null。否则,在调用实例方法时会引发空指针异常。因此,在进行实例方法引用操作前,请先对对象进行必要的空值检查。
  4. 正确使用方法引用,增加代码可读性:正确使用方法引用可以使代码更加简洁、清晰和易于理解。然而,过度使用方法引用可能会导致代码难以阅读和维护。请在合适的情况下使用方法引用,并在需要更明确的表达时使用传统的Lambda表达式(或匿名类)来代替。
  5. 选择最适合的方法引用方式:熟悉各种方法引用的不同形式,包括静态方法引用、实例方法引用、构造方法引用和引用特定类型的任意对象的实例方法。了解这些不同的使用形式,可以根据具体的编码需求选择最适合的方法引用方式。(方法引用不能直接引用抽象方法,也不能引用接口中的默认方法
  6. 方法引用与Lambda表达式的比较:在某些情况下,方法引用可以代替Lambda表达式来提高代码可读性。然而,并非所有情况都适合使用方法引用。有时,使用Lambda表达式可能更清晰明了。

七、优势和缺点

7.1、优点

  • 简化代码:方法引用可以将一些复杂的Lambda表达式转化为简洁明了的代码;
  • 提高代码的复用性:通过方法引用,我们可以重复使用已经存在的方法,而无需重复编写相同的代码。

7.2、缺点 

  • 可读性限制:某些情况下,过多的方法引用可能会降低代码的可读性,特别是当方法名并不直观或者存在多个参数的情况下;
  • 局限性:方法引用只能用于引用非静态方法、静态方法和构造方法(静态方法引用、实例方法引用、构造方法引用和引用特定类型的任意对象的实例方法),对于其他情况则无法使用(不能直接引用抽象方法,也不能引用接口中的默认方法)。

参考文章:【Java基础教程】(二十八)Java新特性篇 · 第八讲:方法引用——概念及优缺点、语法形式及使用案例、应用场景与注意事项 ~_java方法引用重要吗-CSDN博客

【Java 基础篇】Java方法引用详解-CSDN博客

  • 22
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 数组是存储相同类型数据的集合,它们具有固定大小并且在创建后大小不可更改。在Java中,数组通过声明和初始化来创建。声明数组的语法形式如下: ```java int[] arr; // 声明了一个 int 类型的数组 ``` 在声明数组之后,需要通过初始化该数组,也就是为数组分配内存和赋初值。初始化数组的方式有两种:静态初始化和动态初始化。静态初始化是在声明数组的同时给数组元素赋初值的方法,语法形式如下: ```java int[] arr = {1, 2, 3, 4, 5}; // 静态初始化数组 ``` 动态初始化是在声明数组后通过循环或用户输入等方式给数组元素赋值的方法,语法形式如下: ```java int[] arr = new int[5]; // 动态初始化数组 for (int i = 0; i < arr.length; i++) { arr[i] = i + 1; } ``` Java 数组还具有一些常用的属性和方法,如`length`属性用来获取数组的长度,`clone()`方法用来复制数组,`toString()`方法用来将数组转换为字符串等。 除了一维数组外,Java 还支持多维数组,如二维数组、三维数组等。多维数组的声明和初始化方式与一维数组类似,只是需要使用多个`[]`来表示维度。 值得注意的是,Java 中的数组是引用类型,因此在传递数组参数时,实际上传递的是数组的引用,而不是数组的副本。这意味着在方法中对数组的修改会影响到原数组。 总的来说,了解和掌握 Java 数组的声明、初始化、属性和方法,并能灵活运用,对于 Java 编程是非常重要的。希望本文能够为大家提供关于 Java 数组的全面解析和干货知识。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值