《Effective Java》解读第44条:坚持使用标准的函数接口

第44条:坚持使用标准的函数接口

只要标准的函数接口能够满足需求,通常应该优先考虑,而不是专门再构建一个新的函数接口。

函数接口

Java 8 引入 java.util.function 包,里面定义了大量函数式接口,比如:

接口 函数签名 范例
UnaryOperator<T> T apply(T t) String::toLowerCase
BinaryOperator<T> T apply(T t1, T t2) BigInteger::add
Predicate<T> boolean test(T t) Collection::isEmpty
Function<T,R> R apply(T t) Arrays::asList
Supplier<T> T get() Instant::now
Consumer<T> void accept(T t) System.out::println

以及一些其他的变体,所有变体接口的命名都遵循「功能前缀 + 类型标识 + 基础接口名」的组合逻辑。

组成部分 含义 & 常见取值 示例
基础接口名 保留核心功能:Function/Predicate/Consumer/Supplier/Operator Function(函数)、Consumer(消费)
类型标识 标注处理的基本类型:Int/Long/Double(仅这三种,覆盖绝大多数场景) Int(处理 int)、Double(处理 double)
功能前缀 标注「方向 / 参数数量」: 1. ToXXX:表示 "返回 XXX 类型" 2. ObjXXX:表示 "泛型 + XXX 类型" 3. Bi:表示 "双参数" ToInt(返回 int)、ObjLong(泛型 + long)、Bi(双参数)

为什么要坚持使用?

  1. 减少学习成本
    如果你写一个方法,参数类型是自定义的 MyTransformer<T, R>,其他开发者看到后必须先理解这个接口的用途。
    但如果直接用 Function<T, R>,任何人一看就知道是"输入 T,输出 R",无需额外文档。
  2. 提高互操作性与组合能力
java 复制代码
// 不推荐
public interface StringProcessor {
    String process(String s);
}

// 推荐
public void handle(Function<String, String> processor) { ... }

// 使用
handle(String::toUpperCase);
  1. 避免接口爆炸
    如果一个库或应用为每种转型、每种参数数量都自定义函数接口,会导致大量几乎重复的接口,增加认知负担。

什么时候不使用?

  1. 需要多个参数

    比如 (T, U, V) -> R,而标准接口只有 Function(一个参数)和 BiFunction(两个参数)。超过两个参数时,要么自定义,要么用更不直观的方式(如用元组或类包装)。

  2. 需要受检异常

    标准函数接口的方法(如 apply)都不允许抛出受检异常。如果你的操作可能抛出 IOException 等,自定义接口可以声明 throws Exception,更方便处理。

  3. 语义更明确

    例如 java.util.Comparator 就是一个很好的自定义函数接口。虽然功能上它可以用 BiFunction<T, T, Integer> 代替,但 Comparator 表达了"比较器"这一明确语义,并提供了 thenComparing、reversed 等丰富方法。

  4. 需要为接口添加额外的方法

    标准接口只有单个抽象方法,但如果你需要多个默认方法或静态方法来增强用途(如 Comparator 那样),自定义接口更合适。

总结

  1. 优先使用 java.util.function 中的标准函数接口,能让 API 更易读、更通用、更方便与其他 Java 特性协作。

  2. 只在有明确理由(参数数量 > 2、需要受检异常、需要更丰富的 API 或更强的语义)时才自定义函数式接口。

  3. 自定义时,接口命名要体现其用途(如 Comparator、ToLongFunction 等),并标注 @FunctionalInterface。

相关推荐
阿里加多11 小时前
第 4 章:Go 线程模型——GMP 深度解析
java·开发语言·后端·golang
likerhood12 小时前
java中`==`和`.equals()`区别
java·开发语言·python
小小李程序员12 小时前
Langchain4j工具调用获取不到ThreadLocal
java·后端·ai
zs宝来了12 小时前
AQS详解
java·开发语言·jvm
telllong13 小时前
Python异步编程从入门到不懵:asyncio实战踩坑7连发
开发语言·python
wjs202415 小时前
JavaScript 条件语句
开发语言
lulu121654407815 小时前
Claude Code Harness架构技术深度解析:生产级AI Agent工程化实践
java·人工智能·python·ai编程
阿里加多15 小时前
第 1 章:Go 并发编程概述
java·开发语言·数据库·spring·golang
一 乐15 小时前
物流信息管理|基于springboot + vue物流信息管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·物流信息管理系统