@FunctionalInterface

本文介绍Java 8中的函数式接口及其注解@FunctionalInterface,并演示如何使用Lambda表达式简化函数式接口的实例化过程。此外,还列举了几个内置函数式接口的应用场景。

1、理解函数式接口

JDK 1.8 API包含了很多内建的函数式接口,在老Java中常用到的比如Comparator、Runnable、Callable接口,都增加了@FunctionalInterface注解,以便能用在lambda上。

@FunctionalInterface
public interface Runnable {
    /**
     * When an object implementing interface <code>Runnable</code> is used
     * to create a thread, starting the thread causes the object's
     * <code>run</code> method to be called in that separately executing
     * thread.
     * <p>
     * The general contract of the method <code>run</code> is that it may
     * take any action whatsoever.
     *
     * @see     java.lang.Thread#run()
     */
    public abstract void run();
}

FunctionalInterface 的实现如下:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface FunctionalInterface {}

这个注解,是一个信息型注解。
1、只能用在接口上
2、使用此注解的接口,有且仅有一个抽象方法。
3、使用此注解的接口,可以有静态方法和默认方法。
4、使用此注解的接口,可以重写Object中方法,比如equals,toString。
5、该注解不是必须的,如果一个接口符合"函数式接口"定义,那么加不加该注解都没有影响。加上该注解能够更好地让编译器进行检查,如果编写的不是函数式接口,但是加上了@FunctionInterface,那么编译器会报错。

一个正确的函数式接口写法如下:

@FunctionalInterface
public interface FunInterface {
	public Integer method(int x, int y);
	// java.lang.Object中的方法不是抽象方法
	public boolean equals(Object obj);
	// default不是抽象方法
	public default void defaultMethod() {
	}
	// static不是抽象方法
	public static void staticMethod() {
	}
}

2、使用方法

除了可以和普通接口一样写Impl之外,还可通过Lambda表达式进行构造,而不用写Impl class。

FunInterface fun = (a, b) -> {
	return a + b;
};

System.out.print(fun.method(11, 2));

集合遍历与Lambda表达式

List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 0);

System.out.println("---------------------------for循环----------------------\n");
for (Integer i : list) {
    System.out.println(i);
}

System.out.println("---------------------------forEach----------------------\n");
list.forEach(new Consumer<Integer>() {
    @Override
    public void accept(Integer integer) {
        System.out.println(integer);
    }
});

System.out.println("------------------lambda创建函数式接口--------------\n");
list.forEach(i -> System.out.println(i));

集合排序与Lambda表达式

System.out.println("--------------------------传统排序-----------------------\n");
Collections.sort(list, new Comparator<Integer>() {
    @Override
    public int compare(Integer o1, Integer o2) {
        return o2.compareTo(o1);
    }
});

System.out.println("--------------------------lambda排序-----------------------\n");
Collections.sort(list, (Integer o1, Integer o2) -> {
    return o2.compareTo(o1);
});
Collections.sort(list, (o1, o2) -> {
    return o2.compareTo(o1);
});

3、java.util.function

nametypedescriptionmethod
ConsumerConsumer< T >接收T对象,不返回值void accept(T t)
PredicatePredicate< T >接收T对象并返回booleanboolean test(T t);
FunctionFunction< T, R >接收T对象,返回R对象R apply(T t)
SupplierSupplier< T >提供T对象(例如工厂),不接收值T get();
  1. Consumer
    消费某个对象。常用Iterable接口的forEach方法需要传入Consumer,大部分集合类都实现了该接口,用于返回Iterator对象进行迭代。
  2. Predicate
    判断对象是否符合某个条件。比如ArrayList的removeIf(Predicate),可以自定义条件,删除相应的元素。
  3. Function
    实现一个”一元函数“,即传入一个值经过函数的计算返回另一个值。比如 HashMap.computeIfAbsent方法,如果指定的Key未与值关联,使用函数返回值替换。
  4. Supplier
    创建一个对象(工厂类)比如Optional.orElseGet(Supplier<? extends T>):当this对象为null,就通过传入supplier创建一个T返回。
<think>根据给定的引用内容,用户询问的是@FunctionalInterface注解的使用方法。我将从引用中提取相关信息,并按照系统指令要求组织回答。 关键信息提取: - 引用[1]:@FunctionalInterface用于表明接口是函数式接口(符合Java 8+) - 引用[3]:被注解标记的接口必须有且仅有一个抽象方法(覆盖Object类方法不计入) - 引用[4]:函数式接口本质是只有一个抽象方法的接口,关注方法签名而非实现 回答结构: 1. 注解作用:明确说明@FunctionalInterface的核心用途 2. 使用条件:详细说明满足函数式接口的条件 3. 使用方法:给出具体代码示例 4. 注意事项:补充关键使用细节 代码示例将包含: - 正确声明函数式接口 - 使用lambda表达式实现接口 - 错误用法示例(如多个抽象方法) 最后生成相关问题 </think>### @FunctionalInterface注解的使用方法 `@FunctionalInterface` 是 Java 8 引入的注解,用于**标识函数式接口**。函数式接口是指**仅包含一个抽象方法**的接口(覆盖 `Object` 类的方法不计入抽象方法计数)。以下是具体使用方法: --- #### 1. **基本语法** ```java @FunctionalInterface public interface MyFunction { // 唯一的抽象方法 void execute(String input); } ``` - 必须满足条件:**仅有一个抽象方法**(非默认/静态方法)。 - 若违反此条件(如声明多个抽象方法),编译器会报错: ```java @FunctionalInterface public interface InvalidFunction { void method1(); // 抽象方法1 void method2(); // 抽象方法2 → 编译错误! } ``` --- #### 2. **覆盖 `Object` 类的方法不计数** 若接口包含覆盖 `Object` 类的方法(如 `toString()`),它不计入抽象方法数量: ```java @FunctionalInterface public interface Converter { String convert(int num); // 唯一的抽象方法 // 覆盖Object类的方法 → 合法 @Override String toString(); } ``` --- #### 3. **与Lambda表达式结合使用** 函数式接口的核心价值是支持 **Lambda表达式** 或 **方法引用**: ```java // 定义函数式接口 @FunctionalInterface interface MathOperation { int calculate(int a, int b); } public class Main { public static void main(String[] args) { // Lambda实现接口 MathOperation add = (a, b) -> a + b; System.out.println(add.calculate(3, 5)); // 输出8 } } ``` --- #### 4. **常见应用场景** - **简化代码**:替代匿名内部类。 ```java Runnable task = () -> System.out.println("Running"); ``` - **Stream API**:与 `map()`, `filter()` 等方法结合。 ```java List<Integer> numbers = Arrays.asList(1, 2, 3); numbers.stream().filter(n -> n % 2 == 0).forEach(System.out::println); ``` --- #### 5. **注意事项** - **非强制使用**:即使不添加 `@FunctionalInterface`,满足条件的接口仍是函数式接口。但添加注解可确保编译器检查抽象方法数量[^3]。 - **兼容性**:若接口继承其他接口,需确保合并后仍只有一个抽象方法: ```java interface Parent { void parentMethod(); } @FunctionalInterface interface Child extends Parent { // 错误:包含Parent的parentMethod → 共两个抽象方法! } ``` --- ### 相关问题 1. `@FunctionalInterface` 注解在 Java 8 的 Stream API 中如何应用? 2. 如何处理函数式接口与默认方法的冲突? 3. `@FunctionalInterface` 和 Java 的 `@Override` 注解有何设计差异? [^1]: 【Java 基础】-- Java 接口中的 @Public 和 @FunctionalInterface 注解详解 [^3]: java8 functionalinterface注解_@FunctionalInterface注解的使用 [^4]: @FunctionalInterface注解分析及使用
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值