(原)
Java 8 新特性1-函数式接口
Lambda表达式基本结构:
(param1,param2,param3) -> {代码块}
Lambda表达式结构:
(type1 arg1,type2 arg2) -> {body}; //type1、type2表示参数类型,arg1、arg2表示参数,body表示方法体。
(arg1,arg2) -> (body;); //这种写法,编译器会自动判断出arg1,arg2的参数类型,也是正确的;
arg1 -> {body;}; //如果只有一个参数,参数的括号也可以省去。
arg1 -> {return arg1;}; //如果有返回值,可以这么写。
arg1 -> arg1; //输出值只有一句,方法体的大括号可以省掉。
例1:
package com.demo.jdk8; import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer; public class Test2 {
public static void main(String[] args) {
for_test();
for_newMethod();
for_lambda();
} public static void for_test(){
List<Integer> list = Arrays.asList(1,2,3,4,5,6); for(Integer i : list){
System.out.println(i);
}
} public static void for_newMethod(){
List<Integer> list = Arrays.asList(1,2,3,4,5,6); list.forEach(new Consumer<Integer>() { @Override
public void accept(Integer t) {
System.out.println(t);
} });
} public static void for_lambda(){
List<Integer> list = Arrays.asList(1,2,3,4,5,6);
list.forEach(i -> System.out.println(i));
}
}
这三个方法最后执行后的结果都一样
在for_newMethod方法中,可以看到list新增了一个新方法,forEach,可以迭代里面的元素,它的参数是一个consumer接口。
Consumer接口在java.util.function包下,该包是java8新引入的工具包,该接口上有一个@FunctionalInterface注解。
/*
* Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved.
* ORACLE PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.
*/ package java.lang; import java.lang.annotation.*; /**
* An informative annotation type used to indicate that an interface
* type declaration is intended to be a <i>functional interface</i> as
* defined by the Java Language Specification.
*这是一个通知性的注解类型,用于去表示某一个接口声明,它指在规定一个函数式接口,
*这个接口由JAVA语言规范去定义的。(也就是说如果一个接口被加上@FunctionalInterface,
*则表示这个接口是一个函数式接口)
* Conceptually, a functional interface has exactly one abstract
* method. Since {@linkplain java.lang.reflect.Method#isDefault()
* default methods} have an implementation, they are not abstract. If
* an interface declares an abstract method overriding one of the
* public methods of {@code java.lang.Object}, that also does
* <em>not</em> count toward the interface's abstract method count
* since any implementation of the interface will have an
* implementation from {@code java.lang.Object} or elsewhere.
*从概念上来说,一个函数式接口只有一个抽象方法,由于java.lang.reflect.Method的方法sDefault()有一个实现,它们不是抽象的。如果一个接口声明了一个接口的方法,重写了java.lang.Object类里面的一个public方法,那么它也没有计入接口的抽象方法加1,
*因为接口的任意一个实现都会有一个来自于java.lang.Object类或其它地方的一个实现。
* <p>Note that instances of functional interfaces can be created with
* lambda expressions, method references, or constructor references.
*注意,含数式接口的实例可以通过lambda表达式、方法引用、或者构造方法引用来创建。
* <p>If a type is annotated with this annotation type, compilers are
* required to generate an error message unless:
*如果一个注解类型带上了这个@FunctionalInterface,编译器会生成一个错误消息,除非以下几种情况:
* <ul>
* <li> The type is an interface type and not an annotation type, enum, or class.
* <li> The annotated type satisfies the requirements of a functional interface.
* </ul>
*这个类型是一个接口类型,并且不是注解、枚举或class类型
*被注解的类型满足了函数式接口的要求
* <p>However, the compiler will treat any interface meeting the
* definition of a functional interface as a functional interface
* regardless of whether or not a {@code FunctionalInterface}
* annotation is present on the interface declaration.
*然而,编译器将会对待满足函数式接口定义的任义的接口,都会把它当成是一个函数式接口而不管是否在它的声明上增加了@FunctionalInterface注解类型。
* @jls 4.3.2. The Class Object
* @jls 9.8 Functional Interfaces
* @jls 9.4.3 Interface Method Body
* @since 1.8
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface FunctionalInterface {}
把@FunctionalInterface(函数式接口)的文档翻译了一下,还是感觉很绕,再此总结一下,
什么是函数式接口:
1、接口被加上了@FunctionalInterface,表示它是一个函数式接口。
2、如果一个接口只有一个抽象方法,那么表示它是一个函数式接口。
3、如果一个接口有一个抽象方法和一些重写了java.lang.Object类公共方法的方法,那么表示它是一个函数式接口。