Java 教程
- Java - 首页
- Java - 概述
- Java - 历史
- Java - 特性
- Java 与 C++
- JVM - Java 虚拟机
- Java - JDK 与 JRE 与 JVM
- Java - Hello World 程序
- Java - 环境设置
- Java - 基本语法
- Java - 变量类型
- Java - 数据类型
- Java - 类型转换
- Java - Unicode 系统
- Java - 基本运算符
- Java - 注释
- Java - 用户输入
- Java - 日期与时间
Java 控制语句
- Java - 循环控制
- Java - 决策制定
- Java - if-else
- Java - switch
- Java - for 循环
- Java - for-each 循环
- Java - while 循环
- Java - do-while 循环
- Java - break
- Java - continue
面向对象编程
- Java - 面向对象编程概念
- Java - 对象与类
- Java - 类属性
- Java - 类方法
- Java - 方法
- Java - 变量作用域
- Java - 构造函数
- Java - 访问修饰符
- Java - 继承
- Java - 聚合
- Java - 多态
- Java - 重写
- Java - 方法重载
- Java - 动态绑定
- Java - 静态绑定
- Java - 实例初始化块
- Java - 抽象
- Java - 封装
- Java - 接口
- Java - 包
- Java - 内部类
- Java - 静态类
- Java - 匿名类
- Java - 单例类
- Java - 包装类
- Java - 枚举
- Java - 枚举构造函数
- Java - 枚举字符串
Java 内置类
Java 文件处理
Java 错误与异常
- Java - 异常
- Java - try-catch 块
- Java - try-with-resources
- Java - 多重 catch 块
- Java - 嵌套 try 块
- Java - finally 块
- Java - throw 异常
- Java - 异常传播
- Java - 内置异常
- Java - 自定义异常
Java 多线程
- Java - 多线程
- Java - 线程生命周期
- Java - 创建线程
- Java - 启动线程
- Java - 加入线程
- Java - 线程命名
- Java - 线程调度器
- Java - 线程池
- Java - 主线程
- Java - 线程优先级
- Java - 守护线程
- Java - 线程组
- Java - 关闭钩子
Java 同步
Java 网络编程
- Java - 网络编程
- Java - Socket 编程
- Java - URL 处理
- Java - URL 类
- Java - URLConnection 类
- Java - HttpURLConnection 类
- Java - Socket 类
- Java - 泛型
Java 集合
Java 接口
Java 数据结构
Java 集合算法
高级 Java
- Java - 命令行参数
- Java - Lambda 表达式
- Java - 发送邮件
- Java - Applet 基础
- Java - Javadoc 注释
- Java - 自动装箱和拆箱
- Java - 文件不匹配方法
- Java - REPL (JShell)
- Java - 多版本 Jar 文件
- Java - 私有接口方法
- Java - 内部类菱形运算符
- Java - 多分辨率图像 API
- Java - 集合工厂方法
- Java - 模块系统
- Java - Nashorn JavaScript
- Java - Optional 类
- Java - 方法引用
- Java - 函数式接口
- Java - 默认方法
- Java - Base64 编码解码
- Java - switch 表达式
- Java - Teeing 收集器
- Java - 微基准测试
- Java - 文本块
- Java - 动态 CDS 归档
- Java - Z 垃圾收集器 (ZGC)
- Java - 空指针异常
- Java - 打包工具
- Java - 密封类
- Java - 记录类
- Java - 隐藏类
- Java - 模式匹配
- Java - 简洁的数字格式化
- Java - 垃圾收集
- Java - JIT 编译器
Java 其他
- Java - 递归
- Java - 正则表达式
- Java - 序列化
- Java - 字符串
- Java - Process API 改进
- Java - Stream API 改进
- Java - 增强型 @Deprecated 注解
- Java - CompletableFuture API 改进
- Java - 流
- Java - 日期时间 API
- Java 8 - 新特性
- Java 9 - 新特性
- Java 10 - 新特性
- Java 11 - 新特性
- Java 12 - 新特性
- Java 13 - 新特性
- Java 14 - 新特性
- Java 15 - 新特性
- Java 16 - 新特性
Java APIs 与框架
Java 类引用
- Java - Scanner
- Java - 数组
- Java - 字符串
- Java - Date
- Java - ArrayList
- Java - Vector
- Java - Stack
- Java - PriorityQueue
- Java - LinkedList
- Java - ArrayDeque
- Java - HashMap
- Java - LinkedHashMap
- Java - WeakHashMap
- Java - EnumMap
- Java - TreeMap
- Java - IdentityHashMap
- Java - HashSet
- Java - EnumSet
- Java - LinkedHashSet
- Java - TreeSet
- Java - BitSet
- Java - Dictionary
- Java - Hashtable
- Java - Properties
- Java - Collection
- Java - Array
Java 有用资源
Java - 函数式接口
**函数式接口**是在 Java 8 中与lambda 表达式和方法引用一起引入的。这三个特性是为了增强Java中的函数式编程,并编写简洁易读的代码。在 Java 8 之前,需要编写大量样板代码来实现基本功能。例如,为了调用一个函数,首先必须创建一个包含所需方法的类,创建一个类实例,然后使用该实例来调用方法,或者使用匿名类及其相应的方法。
使用 lambda 表达式,我们可以避免对具体类和匿名类对象的需要。**函数式接口**更进一步,因为 lambda 表达式可以很容易地实现函数式接口,因为它只需要实现一个方法。
**函数式接口**只有一个功能。例如,具有单个方法 compareTo() 的 Comparable 接口用于比较目的。但它可以有任意数量的默认方法和静态方法。
Java 8 定义了许多函数式接口,这些接口广泛用于 lambda 表达式中。以下是 java.util.Function 包中定义的函数式接口列表。
@FunctionalInterface 注解
从功能上讲,任何只有一个抽象方法的接口都是函数式接口。Java 提供了一个 @FunctionalInterface 注解,用于将接口标记为函数式接口,以便编译器可以检查接口是否为函数式接口。此注解是可选的,主要用于添加编译器检查,并提高代码的可读性和可维护性。
Java 中函数式接口的类型
Java 中主要有四种类型的函数式接口。
Predicate 函数式接口
谓词函数式接口是一种其方法接受一个参数并返回 true 或 false 的接口。谓词函数式接口主要用于比较排序元素或根据应用于传递输入的某些条件过滤值。Java 还为基本类型提供了谓词函数式接口,例如 IntPredicate、DoublePredicate 和 LongPredicate,它们分别只接受 Integer、Double 和 Long。
用法
Predicate predicate = (value) -> value != 0; // or Predicate predicate = (value) -> test(value);
在上面的代码片段中,谓词函数根据传递的值返回 true/false。
示例
在这个例子中,我们使用谓词函数式接口来过滤整数列表中的奇数,借助 lambda 表达式。
package com.tutorialspoint; import java.util.Arrays; import java.util.List; import java.util.function.Predicate; public class Tester { public static void main(String args[]) { List<Integer> numbers = Arrays.asList(1,2,3,4,5,6,7,8); Predicate<Integer> isEvenNumber = n -> n %2 == 0; numbers = numbers.stream().filter(isEvenNumber).toList(); System.out.println(numbers); } }
让我们编译并运行上面的程序,这将产生以下结果:
[2, 4, 6, 8]
Consumer 函数式接口
Consumer 函数式接口是一种其方法接受一个参数但不返回值的接口。Consumer 函数式接口主要用于副作用操作。例如,打印元素,添加问候语等。Consumer 也有其他变体,如 BiConsumer。BiConsumer 函数式接口可以接受两个参数。Java 还为基本类型提供了 consumer 函数式接口,例如 IntConsumer、DoubleConsumer 和 LongConsumer,它们分别只接受 Integer、Double 和 Long。
用法
Consumer consumer = (value) -> System.out.println(value); // Or Consumer consumer1 = System.out::println; // Or Consumer consumer2 = (value) -> accept(value);
示例
在这个例子中,我们使用 consumer 函数式接口来打印整数列表中的所有数字,借助 lambda 表达式和方法引用。
package com.tutorialspoint; import java.util.Arrays; import java.util.List; import java.util.function.Consumer; public class Tester { public static void main(String args[]) { List<Integer> numbers = Arrays.asList(1,2,3,4,5,6,7,8); Consumer<Integer> consumer = (value) -> System.out.println(value); Consumer consumer1 = System.out::println; System.out.println("Printing using consumer functional interface as lambda expression"); numbers.forEach(consumer); System.out.println("Printing using consumer functional interface as method reference"); numbers.forEach(consumer1); } }
让我们编译并运行上面的程序,这将产生以下结果:
Printing using consumer functional interface as lambda expression 1 2 3 4 5 6 7 8 Printing using consumer functional interface as method reference 1 2 3 4 5 6 7 8
Supplier 函数式接口
Supplier 函数式接口是一种其方法没有任何参数并返回值的接口。Supplier 函数式接口主要用于延迟生成值。例如,获取随机数,生成数字序列等。
用法
Supplier supplier = () -> Math.random() * 10; // or Supplier supplier1 = () -> get();
示例
在这个例子中,我们使用 Supplier 函数式接口来获取随机数,借助 lambda 表达式。
package com.tutorialspoint; import java.util.ArrayList; import java.util.List; import java.util.function.Supplier; public class Tester { public static void main(String args[]) { Supplier<Integer> supplier = () -> (int)(Math.random() * 10); List<Integer> randomNumbers = new ArrayList<>(); // generate 10 random numbers for(int i = 0; i< 10; i++) { randomNumbers.add(supplier.get()); } System.out.println(randomNumbers); } }
让我们编译并运行上面的程序,这将产生以下结果:
[0, 8, 8, 8, 8, 5, 7, 5, 5, 9]
Function 函数式接口
函数式接口是指其方法接受一个参数并返回一个值的方法。函数式接口主要用于获取处理后的值。例如,获取元素的平方、修剪字符串值等。Function还有其他变体,例如BiFunction。BiFunction函数式接口可以接受两个参数。Java也为基本数据类型提供了函数式接口,例如IntFunction、DoubleFunction和LongFunction,它们分别只接受Integer、Double和Long。还有两个实用程序接口,UnaryOperator扩展了Function接口,BinaryOperator扩展了BiFunction接口。
用法
Function function = (value) -> Math.random() * 10; // or Function function1 = (value) -> apply(value);
示例
在这个例子中,我们使用函数式接口和lambda表达式来获取数字平方的列表。
package com.tutorialspoint; import java.util.Arrays; import java.util.List; import java.util.function.Function; public class Tester { public static void main(String args[]) { List<Integer> numbers = Arrays.asList(1,2,3,4,5,6,7,8); Function<Integer, Integer> squared = (value) -> value * value; List<Integer> squaredNumbers = numbers.stream().map(squared).toList(); System.out.println(squaredNumbers); } }
让我们编译并运行上面的程序,这将产生以下结果:
[1, 4, 9, 16, 25, 36, 49, 64]
Java 8之前已存在的函数式接口
在Java中,许多现有的接口都被注释为函数式接口,并且可以在lambda表达式中使用。例如:
Runnable − 提供run()方法
Callable − 提供call()方法
ActionListener − 提供actionPerformed()方法
Comparable − 提供compareTo()方法来比较两个数字
示例
在这个例子中,我们创建了两个线程。第一个使用匿名类创建,第二个使用lambda表达式创建。两者都使用Runnable接口来创建线程实例。
package com.tutorialspoint; public class Tester { public static void main(String args[]) { // create anonymous inner class object new Thread(new Runnable() { @Override public void run() { System.out.println("Thread 1 is running"); } }).start(); // lambda expression to create the object new Thread(() -> { System.out.println("Thread 2 is running."); }).start(); } }
让我们编译并运行上面的程序,这将产生以下结果:
Thread 1 is running Thread 2 is running.