Java8新特性之函数式接口

Java8是 Java 语言的一个重要版本,该版本于2014年3月发布,是自Java5以来最具革命性的版本,这个版本包含语言、编译器、库、工具和JVM等方面的十多个新特性。

函数式接口

  • 函数式接口主要指只包含一个抽象方法的接口,如:java.lang.Runnable、java.util.Comparator 等
  • Java8提供@FunctionalInterface注解来定义函数式接口,若定义的接口不符合函数式的规范便会报错
  • Java8中增加了java.util.function包,该包包含了常用的函数式接口
  • 在这里插入图片描述
匿名内部类实现函数式接口
public class FunctionalInterfaceTest {

    public static void main(String[] args) {

        // 匿名内部类的语法格式:父类/接口类型 引用变量名 = new 父类/接口类型(){ 方法的重写 };
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                System.out.println("我是既没有参数又没有返回值的方法!");
            }
        };
        runnable.run(); // 我是既没有参数又没有返回值的方法!
    }
    
    	System.out.println("----------------------------------------------------------------------");
        Consumer consumer = new Consumer() {
            @Override
            public void accept(Object o) {
                System.out.println(o + "有参但没有返回值的方法就是我!");
            }
        };
        consumer.accept("友情提示:"); // 友情提示:有参但没有返回值的方法就是我!
    
    	System.out.println("----------------------------------------------------------------------");
        Supplier supplier = new Supplier() {
            @Override
            public Object get() {
                return "无参有返回值!";
            }
        };
        System.out.println(supplier.get()); // 无参有返回值
}
Lambda表达式实现函数式接口
public class FunctionalInterfaceTest {

    public static void main(String[] args) {

        // 使用lambda表达式实现函数式接口对象的创建: (参数列表)->{方法体;}
        Runnable runnable1 = () -> { System.out.println("我是既没有参数又没有返回值的方法!"); };
        runnable1.run();
        
        System.out.println("----------------------------------------------------------------------");
        //Consumer consumer1 = (Object o) -> {System.out.println(o + "有参但没有返回值的方法就是我!");};
        //Consumer consumer1 = (o) -> System.out.println(o + "有参但没有返回值的方法就是我!");
        // 省略了()、参数类型、{}  自动类型推断
        Consumer consumer1 = o -> System.out.println(o + "有参但没有返回值的方法就是我!");
        consumer1.accept("友情提示:");
        
        System.out.println("----------------------------------------------------------------------");
        //Supplier supplier1 = () -> {return "无参有返回值!";};
        Supplier supplier1 = () -> "无参有返回值!";
        System.out.println(supplier1.get());
    }
}
方法引用实现函数式接口
  • 方法引用主要指通过方法的名字来指向一个方法而不需要为方法引用提供方法体,该方法的调用交给函数式接口执行

  • 方法引用是在特定场景下lambda表达式的一种简化表示,可以进一步简化代码的编写使代码更加紧凑简洁,从而减少冗余代码

  • 方法引用使用一对冒号 :: 将类或对象与方法名进行连接,通常使用方式如下:

    • 对象的非静态方法引用 ObjectName :: MethodName

    • 类的静态方法引用 ClassName :: StaticMethodName

    • 类的非静态方法引用 ClassName :: MethodName

      • 其中一个参数对象作为调用对象来调用方法时,可以使用上述方式

      • Comparator<Integer> comparator3 = new Comparator<Integer>() {
            @Override
            // compare两个参数
            public int compare(Integer o1, Integer o2) {
                // 其中一个参数对象作为调用对象来调用方法
                return o1.compareTo(o2);
            }
        };
        System.out.println(comparator3.compare(10, 20)); // -1
        
        // 可以使用类的非静态方法引用 ClassName :: MethodName 
        // Comparator一个参数
        Comparator<Integer> comparator5 = Integer::compareTo;
        System.out.println(comparator5.compare(10, 20)); // -1
        
    • 构造器的引用 ClassName :: new

    • 数组的引用 TypeName[] :: new

  • 举例:对象的非静态方法引用

    public class Person {
        private String name;
        private int age;
    
        public Person() {
        }
    
        public Person(String name, int age) {
            this.name = name;
            this.age = age;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public int getAge() {
            return age;
        }
    
        public void setAge(int age) {
            this.age = age;
        }
    
        public void show() {
            System.out.println("没事出来秀一下哦");
        }
    
        @Override
        public String toString() {
            return "Person{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    '}';
        }
    }
    
  • 无参无返回值

    public class MethodReferenceTest {
    
        public static void main(String[] args) {
    
            // 1.使用匿名内部类的方式通过函数式接口Runnable中的方法实现对Person类中show方法的调用
            Person person = new Person("zhangfei", 30);
            Runnable runnable = new Runnable() {
                @Override
                public void run() {
                    person.show();
                }
            };
            runnable.run(); // 没事出来秀一下哦
    
            // 2.使用lambda表达式的方式实现Person类中show方法的调用
            Runnable runnable1 = () -> person.show();
            runnable1.run(); // 没事出来秀一下哦
           
            // 3.使用方法引用的方式实现Person类中show方法的调用
            // Person类中show方法和Runnable接口中run方法都是无参无返回值 结构一致 引用容易
            Runnable runnable2 = person::show;
            runnable2.run();
        }
    }
    
  • 有参无返回值

    public class MethodReferenceTest {
    
        public static void main(String[] args) {
            
            // 4.使用匿名内部类的方式通过函数式接口Consumer中的方法来实现Person类中setName方法的调用
            Consumer<String> consumer = new Consumer<String>() {
                @Override
                public void accept(String s) {
                    person.setName(s);
                }
            };
            consumer.accept("guanyu");
            System.out.println("person = " + person); // guanyu 30
            
            // 5.使用lambda表达式的方式实现Person类中setName方法的调用
            Consumer<String> consumer1 = s -> person.setName(s);
            consumer1.accept("liubei");
            System.out.println("person = " + person); // liubei 30
    
            // 6.使用方法引用的方式实现Person类中setName方法的调用
            Consumer<String> consumer2 = person::setName;
            consumer2.accept("zhangfei");
            System.out.println("person = " + person); // zhangfei 30
        }
    }
    
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值