《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。

相关推荐
像我这样帅的人丶你还7 小时前
Java 后端详解(四):分页与搜索
java·javascript·后端
她的男孩7 小时前
数据权限为什么不能只靠注解?Forge 的 Mapper 层 SQL 改写源码拆解
java·后端·架构
tntxia8 小时前
Mybatis的日志输入
java
亦暖筑序9 小时前
Java 8老系统Browser Agent实战:三层拦截把AI操作后台变成可审计流程
java·后端·设计模式
用户2986985301412 小时前
Java 实现 Word 文档加密与权限解除
java·后端
Yeats_Liao13 小时前
14:Servlet中的页面跳转-Java Web
java·后端·架构
未秃头的程序猿13 小时前
告别"if-else地狱"!Java 21模式匹配,代码优雅了10倍
java·后端·面试
鹤望兰67513 小时前
字节跳动国际支付-后端开发-三面面经
java
Flittly14 小时前
【AgentScope Java新手村系列】(14)人机交互
java·spring boot·spring
RainCity14 小时前
Java Swing 自定义组件库分享(十二)
java·笔记·后端