Lambda表达式和函数式接口

初识lambda表达式

lambda表达式Java8的新特性,可以将lambda表达式看成是精简语法的匿名内部类。

举个例子,一般的匿名类处理如下

// 为button注册一个事件 
button.setOnAction( 
     new EventHandler<ActionEvent>(){
         @Override
         public void handle(ActionEvent e){
             // do something
         }
     }
 );

如果使用lambda表达式,代码可以得到极大地简化,如下所示:

button.setOnAction( (e) ->{
    //do something
	}
)

为了更好地理解lambda表达式,我们先引入函数式接口的概念。

函数式接口(functional interface)

简单地说,接口中若只包含一个抽象方法,则称该接口为函数式接口。可以在接口前使用注解@FunctionalInterface检查该接口是否为函数式接口。

lambda表达式基本语法

Java8中,引入了一个新的操作符"->",该操作符将lambda表达式拆分成两个部分:

  • lambda表达式的参数列表,位于"->"操作符左边
  • lambda表达式所执行的功能,即lambda体。位于"->"操作符右边。

一个lambda表达式的基础语法是

(type1 param1, type2 param2, ...)->{
    statements;
}

编译器对待一个lambda表达式就如同它是从一个匿名内部类创建的对象。

简单来说,lambda表达式的参数列表就是函数式接口中抽象方法的参数列表,lambda体就是该方法的方法体。

以本文开头的代码为例,EventHandler接口仅有一个方法,并且该方法具有一个ActionEvent类型的参数,所以编译器可以自动推断出e是一个ActionEvent类型的参数,并且花括号里面的内容为handle方法的方法体。

如果EventHandler接口中含有多个方法,编译器将无法编译lambda表达式,可以看出,lambda表达式是根据编译器的隐式推断来简化代码的。所以,lambda表达式需要函数式接口的支持

明白了lambda表达式的运行机制,下面就是常见的lambda表达式的语法格式:

  1. 函数式接口中的方法无参数,例子如下:

    () ->{//do something}

    若方法体只有一个语句,return和花括号都可以省略不写。

  2. 函数式接口中的方法有参数,例子如下:

    (int x, int y) -> {//do something}

    表达式中参数的类型可以不写,编译器可以通过上下文推断出其类型,上面的代码可以简化如下:

    (x, y) ->{do something}

    如果只有一个参数,参数列表的括号可以省略,例子如下:

    x ->{//do something}

四大核心函数式接口

lambda表达式需要有函数式接口支持,单并不是说每次用lambda表达式时都要自定义一个函数式接口,实际上,Java8已经为我们准备了java.util.function包,其中有许多非常实用的函数式接口,大致可以分为4种,下面介绍4种核心的接口的典型代表。

消费型接口

消费型接口典型的代表为Consumer,其抽象方法为accept(), 该方法仅接受一个参数,并且没有返回值。示例如下:

	//打印字符串
	public static void main(String[] args) {
		handle("Hello world!", (s) -> System.out.println(s));
	}
	public static void handle(String s, Consumer<String> con) {
		con.accept(s);
	}

供给型接口

供给型接口的典型代表为Supplier,其抽象方法为get(), 该方法不接受参数,但有返回值。示例如下:

	public static void main(String[] args) {
		List<Integer> list = getNumList(10, () -> (int)(Math.random() *100));
		for(Integer i:list) {
			System.out.println(i);
		}
	}
	//产生指定个数的整数,并将其置于集合中
	public static List<Integer> getNumList(int num, Supplier<Integer> sup){
		List<Integer> list = new ArrayList<>();
		for(int i = 0; i < num; i++) {
			list.add(sup.get());
		}
		return list;
	}

函数型接口

函数型接口中的代表为Function,其抽象方法位apply(), 接受有一个参数,并且有返回值。示例如下:

	//将字符串转化为整型
	public static void main(String[] args) {
		System.out.println(convert("100", (e) -> Integer.parseInt(e)));
	}
	public static Integer convert(String str, Function<String, Integer> fun) {
		return fun.apply(str);
	}

断言型接口

断言型接口的典型接口为Predicate,其抽象方法为test(), 接受一个参数,并返回一个布尔值。示例如下:

	// 将满足条件的整数放于集合中
	public static void main(String[] args) {
		List<Integer> list = Arrays.asList(1, 2, 3, 5, 7, 11, 20, 47);
		List<Integer> myList = filterInt(list, (e) -> e > 3);
		for(Integer i : myList) {
			System.out.println(i);
		}
	}
	
	public static List<Integer> filterInt(List<Integer> list, Predicate<Integer> pre ){
		List<Integer> myList = new ArrayList<>();
		for(Integer i : list) {
			if(pre.test(i)) {
				myList.add(i);
			}
		}
		return myList;
	}
  • 5
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值