Java8——方法引用和构造器引用

简介: 方法引用是Lambda表达式的另外一种表现形式
日积月累,水滴石穿 😄

什么是方法引用

方法引用是Lambda表达式的另外一种表现形式,是一个语法糖。那为什么要使用方法引用呢?

当 Lambda 体中的具体实现,已经有其他方法帮我们实现过了,那这时候我们就可以使用方法引用。也就是说使用 方法引用 可以少写一些代码,提高工作效率。使用方法引用的时候 需要保证引用方法的参数列表、返回值类型与我们当前所要实现的函数式接口方法的参数列表、返回值类型保持一致

语法格式

对象::实例方法名

 @Test
public void test1(){
    Consumer<String> con = (x) ->System.out.println(x);
    //执行 Lambda 体
    con.accept("gongj");

    //简写
    Consumer<String> con2 = System.out::println;
    con2.accept("yuanj");
}

注意,这样写的前提: accept()方法和println()方法的参数列表和返回类型要完全一致:
image.png
image.png

先准备一个 Employee对象

   public class Employee {
    private String name;
    private int age;
    private double salary;

    public Employee() {
    }
    public Employee(String name, int age, double salary) {
        this.name = name;
        this.age = age;
        this.salary = salary;
    }

    public Employee(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }
    public int getAge() {
        return age;
    }
    public double getSalary() {
        return salary;
    }

    @Override
    public String toString() {
        return "name='" + name + '\'' +
                ", age=" + age +
                ", salary=" + salary;
    }
}

再看一个例子

@Test
public void test2(){

    Employee emp = new Employee("gongj",122,7888);

    Supplier<String> sup = () -> emp.getName();
    System.out.println(sup.get());

    //简写
    Supplier<String> sup2 = emp::getName;
    System.out.println(sup2.get());
}

image.png
image.png

类::静态方法名

@Test
public void test3() {
    Comparator<Integer> com = (x, y) -> Integer.compare(x, y);
    com.compare(1, 2);

    // 使用方法引用的方式
    Comparator<Integer> com1 = Integer::compare;
    com1.compare(1, 2);
}

类::实例方法名

@Test
public void test4(){
    BiPredicate<String,String> bp = (x,y) -> x.equals(y);
    System.out.println(bp.test("gognj","dddd"));

    BiPredicate<String,String> bp2 = String::equals;
    System.out.println(bp.test("gognj","gognj"));
}

使用注意: 若Lambda参数列表中的第一个参数是实例方法的调用者,而第二个参数是实例方法的参数时,可以使用 类::实例方法名。 这种方法引用的方式就不需要满足保证引用方法的参数列表、返回值类型与我们当前所要实现的函数式接口方法的参数列表、返回值类型保持一致这一规则了
image.png
image.png

构造器引用

语法格式:类::new,调用哪个构造器取决于函数式接口中的方法形参的定义,Lambda 会自动根据接口方法推断出你要调用的构造器,也就是说 需要调用的构造器的参数列表要与函数式接口中的抽象方法的参数列表保持一致

  • 调用无参构造器
@Test
public void test5(){
   Supplier<Employee> sup = () -> new Employee();
   
   //构造器引用  这里调用的是无参构造器
   Supplier<Employee> sup2 = Employee::new;
   System.out.println(sup2.get());
}
结果:
name='null', age=0, salary=0.0

image.png
image.png

  • 调用有参构造器
@Test
public void test6(){
    Function<String,Employee> fun = (x) -> new Employee(x);
    System.out.println(fun.apply("gongj"));

    Function<String,Employee> fun2 = Employee::new;
    System.out.println(fun2.apply("yuanj"));
}
结果:
name='gongj', age=0, salary=0.0
name='yuanj', age=0, salary=0.0

image.png
image.png

如果想要匹配多个的,(两个的可以使用BiFunction),下面看一个三个参数的,自定义一个函数式接口:

@FunctionalInterface
public interface MyFun2<T,A,B, R> {

    R apply(T t,A a,B b);
}

测试

 @Test
    public void test7(){

        MyFun2<String,Integer, Double,Employee> fun = Employee::new;
        System.out.println(fun.apply("gongj",99999d));
    }
结果:
name='gongj', age=25, salary=99999.0

数组引用

语法格式为 Type[]::new

    @Test
    public void test8() {
        Function<Integer, String[]> fun = (x) -> new String[x];
        String[] apply = fun.apply(10);
        System.out.println(apply.length);

        // 数组引用
        Function<Integer, String[]> fun1 = String[]::new;
        String[] apply2 = fun1.apply(10);
        System.out.println(apply2.length);
    }
结果:
10
20
  • 如你对本文有疑问或本文有错误之处,欢迎评论留言指出。如觉得本文对你有所帮助,欢迎点赞和关注。
相关文章
|
5天前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
15 1
|
2月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
48 17
|
23天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
76 4
|
28天前
|
Java 测试技术 Maven
Java一分钟之-PowerMock:静态方法与私有方法测试
通过本文的详细介绍,您可以使用PowerMock轻松地测试Java代码中的静态方法和私有方法。PowerMock通过扩展Mockito,提供了强大的功能,帮助开发者在复杂的测试场景中保持高效和准确的单元测试。希望本文对您的Java单元测试有所帮助。
73 2
|
2月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
20 3
|
2月前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
19 2
|
2月前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
19 1
|
2月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
36 1
|
2月前
|
Java
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅。它们用于线程间通信,使线程能够协作完成任务。通过这些方法,生产者和消费者线程可以高效地管理共享资源,确保程序的有序运行。正确使用这些方法需要遵循同步规则,避免虚假唤醒等问题。示例代码展示了如何在生产者-消费者模型中使用`wait()`和`notify()`。
29 1
|
29天前
|
Java Spring
JAVA获取重定向地址URL的两种方法
【10月更文挑战第17天】本文介绍了两种在Java中获取HTTP响应头中的Location字段的方法:一种是使用HttpURLConnection,另一种是使用Spring的RestTemplate。通过设置连接超时和禁用自动重定向,确保请求按预期执行。此外,还提供了一个自定义的`NoRedirectSimpleClientHttpRequestFactory`类,用于禁用RestTemplate的自动重定向功能。