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 的父类
相关推荐
二进制person6 小时前
JavaEE初阶 --文件操作和IO
java·java-ee
@老蝴6 小时前
Java EE - 线程安全的产生及解决方法
java·开发语言·java-ee
せいしゅん青春之我6 小时前
【JavaEE初阶】网络层-IP协议
java·服务器·网络·网络协议·tcp/ip·java-ee
Han.miracle6 小时前
Java ee初阶——定时器
java·java-ee
飞鱼&7 小时前
HashMap相关问题详解
java·hashmap
白露与泡影7 小时前
面试:Spring中单例模式用的是哪种?
spring·单例模式·面试
没有bug.的程序员7 小时前
Spring Cloud Alibaba 生态总览
java·开发语言·spring boot·spring cloud·alibaba
快乐非自愿8 小时前
Java垃圾收集器全解:从Serial到G1的进化之旅
java·开发语言·python
树在风中摇曳8 小时前
Java 静态成员与继承封装实战:从报错到彻底吃透核心特性
java·开发语言
芳草萋萋鹦鹉洲哦11 小时前
【Windows】tauri+rust运行打包工具链安装
开发语言·windows·rust