Java Lambda 类型推断详解:filter() 方法与 Predicate<? super T>

一、问题核心解析

1. 代码示例分析

复制代码
List<String> strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl");
List<String> filtered = strings.stream()
        .filter(string -> !string.isEmpty()) 
        .collect(Collectors.toList());

2. Lambda 表达式为什么能传入

  • 函数式接口本质:Lambda 表达式是函数式接口的实例化简写

  • Predicate<T> 是函数式接口,只有一个抽象方法:

    复制代码
    @FunctionalInterface
    public interface Predicate<T> {
        boolean test(T t); // 唯一的抽象方法
        // 其他默认方法...
    }
  • Lambda 表达式 string -> !string.isEmpty() 等价于:

    复制代码
    new Predicate<String>() {
        @Override
        public boolean test(String string) {
            return !string.isEmpty();
        }
    }

3. <? super T> 的含义

  • 下界通配符 :表示接受 TT 的父类型

  • filter(Predicate<? super T> predicate) 中:

    • T 是流元素的类型(本例中是 String

    • ? super T 表示可以接受处理 String 或其父类(如 Object, CharSequence)的 Predicate

二、类型推断全流程

1. 类型推断步骤

2. 详细推断过程

步骤 1:流类型确定
复制代码
strings.stream() // 返回 Stream<String>
  • 编译器从 strings 的类型 List<String> 推断出 stream() 返回 Stream<String>

  • 此时 T = String

步骤 2:filter 方法签名匹配
复制代码
Stream<T> filter(Predicate<? super T> predicate);
// 代入 T=String → 
Stream<String> filter(Predicate<? super String> predicate);
步骤 3:Lambda 表达式类型解析
复制代码
.filter(string -> !string.isEmpty())
  • 目标类型Predicate<? super String>

  • 参数类型推断

    • 根据目标类型,string 必须是 String 或其父类

    • Lambda 体中使用 string.isEmpty() 方法

    • isEmpty()String 类的方法(在 CharSequence 中不存在)

    • 因此编译器推断 string 必须是 String 类型

步骤 4:返回值类型推断
  • Lambda 体 !string.isEmpty() 返回 boolean

  • 匹配 Predicate.test() 要求的 boolean 返回类型

步骤 5:完整匹配验证
复制代码
// 等效实现
Predicate<String> p = (String s) -> {
    return !s.isEmpty();
};
filter(p); // 符合 Predicate<? super String>

三、<? super T> 的设计意义

1. 提高 API 灵活性

复制代码
// 通用谓词:处理任何对象
Predicate<Object> nonNull = obj -> obj != null;

// 可用于不同类型的流
List<String> strings = ...;
strings.stream().filter(nonNull); // 允许:Object 是 String 的父类

List<Integer> numbers = ...;
numbers.stream().filter(nonNull); // 允许:Object 是 Integer 的父类
相关推荐
大飞pkz16 分钟前
【设计模式】责任链模式
开发语言·设计模式·c#·责任链模式
gplitems12341 分钟前
Gunslinger – Gun Store & Hunting WordPress Theme: A Responsible
开发语言·前端·javascript
fly-phantomWing41 分钟前
Maven的安装与配置的详细步骤
java·后端·maven·intellij-idea
大飞pkz2 小时前
【设计模式】六大基本原则
开发语言·设计模式·c#·六大原则
iCxhust2 小时前
Intel8259汇编串口接收转C语言
c语言·开发语言·汇编
掘根3 小时前
【Qt】布局管理器
开发语言·qt
半夏知半秋3 小时前
skynet-socket.lua源码分析
服务器·开发语言·学习·架构·lua
2401_841495644 小时前
【数据结构】红黑树的基本操作
java·数据结构·c++·python·算法·红黑树·二叉搜索树
西猫雷婶4 小时前
random.shuffle()函数随机打乱数据
开发语言·pytorch·python·学习·算法·线性回归·numpy
学编程的小鬼4 小时前
SpringBoot 自动装配原理剖析
java·spring boot·后端