Java8引入了Lambda表达式,可以不必编写FunctionalInterface的实现类,直接写Lambda表达式。除了Lambda表达式,我们还可以直接传入方法引用

方法引用是指:如果某个方法签名和接口恰好一致,可以直接传入方法引用,格式:类名::方法名

静态方法引用

在SortedBy这个类中定义了一个静态方法,其方法签名是传入2个String,返回int。

class SortedBy{
static int nameIgnoreCase(String s1, String s2){
return s1.toLowerCase.compareTo(s2.toLowerCase());
}
}

此时和范型的Comparator接口比较,这2个方法签名是一致的。

注意:这里的方法签名只看方法类型和返回值类型。

```#java
interface Comparator{
int compare(String s1, String s2);
}
```
ignoreCase恰好符合Comparator的接口定义。因此在需要传入Comparator的地方,我们直接传入方法引用
```#java
public static void main(String[] args){
String[] array = "Java Apple lambda functional OOP".split(" ");
Arrays.sort(array,SortedBy::nameIgnoreCase);
System.out.println(Arrays.toString(array));
}
```

实例方法引用

public class String{
public int compareToIgnoreCase(String s){
...
}
}

String的compareToIgnoreCase参数是String,返回值是int,看上去并不符合Comparator接口的定义。

但是实例方法调用时,必须有一个隐含的实例变量。即调用String类的compareToIgnoreCase必须有一个String类型的实例,实例参数+方法参数就通Comparator接口定义是一致的。

注意:隐含的this实例变量是方法的第一个参数。

Arrays.sort(array, String::compareToIgnoreCase);
//=>instance.compareToIgnoreCase(s)
//=> Comparator.compare(instance, s)

构造方法引用

class Person{
String name;
public Person(String name){
this.name = name;
}
}

把一个String类型的List转化为Person类型的List?

方法1:先定义一个ArrayList,然后用for循环来填充这个List。

    List<String> names = Arrays.asList("Bob", "Alice", "Tim");
List<Person> person = new ArrayList<>();
for(String name: names){
Person p = new Person(name);
persons.add(p);
}

方法2:直接引用Person类的构造方法,不用手动创建Person实例

map()传入的方法签名:参数为String,返回值为Person对象。编译器自动查找符合条件的构造方法

构造方法的引用为:类名::new

虽然Java构造方法内部不能持有return语句,但构造方法是有返回值的,它的返回值是this,即当前实例。

    List<String> names = Arrays.asList("Bob", "Alice", "Tim");
List<Person> persons = names.stream().map(Person::new).collect(collectors.toList());
import java.util.Arrays;

class SortedBy {
static int name(String s1, String s2) {
return s1.compareTo(s2);
} static int nameIgnoreCase(String s1, String s2) {
return s1.toLowerCase().compareTo(s2.toLowerCase());
} static int length(String s1, String s2){
int n1 = s1.length();
int n2 = s2.length();
if(n1==n2){
return s1.compareTo(s2);
}
return n1 < n2 ? -1:1;
}
}
public class LambdaSort {
public static void main(String[] args){
String[] array = "Java Apple lambda functional OOP".split(" ");
Arrays.sort(array,SortedBy::nameIgnoreCase);
System.out.println(Arrays.toString(array));
}
}

总结

Functional Interface可以传入:

  • 接口的实现类(代码较繁琐)
  • Lambda表达式
  • 符合方法签名的静态方法
  • 符合方法签名的实例方法(实例类型被看作第一个参数类型)
  • 符合方法签名的构造方法(实例类型被看作返回类型)

廖雪峰Java16函数式编程-1Lambda表达式-3方法引用的更多相关文章

  1. 廖雪峰Java16函数式编程-1Lambda表达式-1Lambda基础

    1. 函数式编程 Java有2类方法: 实例方法:通过实例调用 静态方法:通过类名调用 Java的方法相当于过程式语言的函数 函数式编程(Functional Programing): 把函数作为基本 ...

  2. 廖雪峰Java16函数式编程-2Stream-7其他操作

    1. 排序 Stream<T> sorted(); //按元素默认大小排序(必须实现Comparable接口) Stream<T> sorted(Comparator<? ...

  3. 廖雪峰Java16函数式编程-2Stream-6reduce

    1. 聚合方法 Stream.reduce()是一个Stream的聚合方法:把一个Stream的所有元素聚合成一个结果 例如: Stream.of(1, 2, 3, 4, 5).count(); // ...

  4. 廖雪峰Java16函数式编程-2Stream-5filter

    1.filter简介 Stream.filter()是一个转换方法,把一个Stream转换为另一个Stream. 所谓filter操作,就是对一个Stream的所有元素进行测试,不满足条件的元素就被过 ...

  5. 廖雪峰Java16函数式编程-2Stream-4map

    1. map()简介 Stream.map()是一个Stream的转换方法,把一个stream转换为另一个Stream,这2个Stream是按照映射函数一一对应的. 所谓map操作,就是把一种操作运算 ...

  6. 廖雪峰Java16函数式编程-2Stream-2创建Stream

    1. 方法1:把一个现有的序列变为Stream,它的元素是固定的 //1.直接通过Stream.of()静态方法传入可变参数进行创建 Stream<Integer> s = Stream. ...

  7. 廖雪峰Java16函数式编程-2Stream-1Stream简介

    1. Stream Java8引入全新的Stream API 位于java.util.stream包 1.1 Stream API不同于java.io的InputStream/OutputStream ...

  8. [python学习篇][廖雪峰][2]函数式编程

    函数名也是变量: >>> f = abs >>> f(-10) 10 然变量可以指向函数,函数的参数能接收变量,那么一个函数就可以接收另一个函数作为参数,这种函数就 ...

  9. Java中的函数式编程(四)方法引用method reference

    写在前面 我们已经知道,lambda表达式是一个匿名函数,可以用lambda表达式来实现一个函数式接口.   很自然的,我们会想到类的方法也是函数,本质上和lambda表达式是一样的,那是否也可以用类 ...

随机推荐

  1. 2019基于python的网络爬虫系列,爬取糗事百科

    **因为糗事百科的URL改变,正则表达式也发生了改变,导致了网上许多的代码不能使用,所以写下了这一篇博客,希望对大家有所帮助,谢谢!** 废话不多说,直接上代码. 为了方便提取数据,我用的是beaut ...

  2. android Intent和IntentFilter

    android的应用程序包含三种重要的组件:Activity.Service.BroadcastReceiver,应用程序采用一致的方式来启动他们——都是依靠Intent来进行启动.Intent就封装 ...

  3. Linux --赋予普通用户root 权限

    Linux的普通用户在安装一些东西的时候或者执行命令的时候,终端始终会提示权限不够,我们会将这个普通用户赋予root权限,但是,和root还是有区别的,因为只能执行root规定好的一些操作命令. 1. ...

  4. 49. ArrayList LinkedList中特有的方法

    集合的体系:--------------| Collection  单列集合的根接口 ----------| List 如果实现了List接口的集合类,该类具备的特点是:有序,可重复 ------|A ...

  5. python 获取年月日时分秒 获取当前时间 datetime函数

    import datetime#取当前时间print(datetime.datetime.now())#取年print(datetime.datetime.now().year)#取月print(da ...

  6. Linux统计根分区使用率

    #!/bin/bash # 统计根分区使用率 # 作者: shaohsiung # 时间: // rate=$(df -h | grep "dev/sda3" | awk '{pr ...

  7. twentytwenty插件,图片对比轮播

    https://zurb.com/playground/twentytwenty 项目应用 http://decortrim.mml.digital/

  8. shell 脚本定制与重定向

    脚本定制 . 或者 source: 读取文本文件并执行(在当前shell解释并执行) source ./ld 总用量 8 -rw-------. 1 root root 1223 10月 2 21:1 ...

  9. unity 打包Apk生成签名证书keystore

    进行Android项目开发中想要将androidapp导出为apk的时候需要选择一个数字证书,即keystore文件(android.keystore),它用来对我们的APP进行签名,是导出APP的一 ...

  10. thinkphp session支持

    系统提供了Session管理和操作的完善支持,全部操作可以通过一个内置的session函数完成,该函数可以完成Session的设置.获取.删除和管理操作. session初始化设置 如果session ...