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

相关推荐
宝耶2 小时前
Java面试题5:List、Set、Map 的区别?各自有哪些实现类?
java·开发语言·list
刘 大 望2 小时前
MCP详细介绍以及IDE和Spring AI中应用
java·ide·人工智能·spring·ai·aigc·ai编程
yunyun321232 小时前
动态库热加载技术
开发语言·c++·算法
88号技师2 小时前
2026年3月一区SCI-B样条曲线优化算法B-spline curves optimizer-附Matlab免费代码
开发语言·算法·数学建模·matlab·优化算法
dapeng28702 小时前
C++中的享元模式实战
开发语言·c++·算法
毕设源码-朱学姐2 小时前
【开题答辩全过程】以 基于springBoot的考试成绩管理系统为例,包含答辩的问题和答案
java·spring boot·后端
jing-ya2 小时前
day 60 图论part11
java·数据结构·算法·图论
常利兵2 小时前
Java后端定时任务抉择:@Scheduled、Quartz、XXL - Job终极对决
java·数据库·sql
程序员爱酸奶2 小时前
Java后端工程师成长指南
java·开发语言