为了账号安全,请及时绑定邮箱和手机立即绑定

Java8方法引用

标签:
Java

内容简介:
方法引用Demo详解
通过5种语法使用方法引用
方法引用使用总结

  1. 在Java8中方法引用Demo详解
    1.1 方法引用出现的背景
    在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。
    那么考虑一种情况:如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑呢?
    1.2 问题的引出

   //函数式接口,用于打印字符串.    @FunctionalInterface    interface Print{        public void print(String s);    }    //使用lambda表达式完成案例测试.    public class PrintDemo {        public static void main(String[] args) {            //Lambda方式解决            print(s -> {System.out.println(s);}, "hello");        }        public static void print(Print p,String s){            p.print(s);        }    }    输出结果:            hello

1.3 问题的发现与解决
这段代码的问题在于,对String进行控制台打印输出的操作方案,明明已经有了现成的实现,那就是System.out对象中的println(String)方法。既然Lambda希望做的事情就是调用println(String)方法,那何必自己手动调用呢?
能否省去Lambda的语法格式(尽管它已经相当简洁)呢?只要“路由”过去就好了:

//函数式接口,用于打印字符串.    @FunctionalInterface    interface Print{        public void print(String s);    }    //使用lambda表达式完成案例测试.    public class PrintDemo {        public static void main(String[] args) {            //方法引用解决方式            //请注意其中的双冒号“::”写法,这被称为“方法引用”,而双冒号是一种“引用运算符”。            print(System.out :: print,"world");        }        public static void print(Print p,String s){            p.print(s);        }    }    输出结果:            world

1.4 方法引用案例的总结
以上例中,System.out对象中有一个重载的println(String)方法恰好就是我们所需要的。那么对于printString方法的函数式接口参数,对比下面两种写法:
Lambda表达式:s -> System.out.println(s);
方法引用:System.out::println
第一种语义是指:拿到参数之后经Lambda之手,继而传递给System.out.println方法去处理。第二种等效写法的语义是指:直接让System.out中的println方法来取代Lambda。两种写法的执行效果完全一样,而第二种方法引用的写法复用了已有方案,更加简洁。1.5 引用运算符
双冒号“::”为引用运算符,而它所在的表达式被称为方法引用。如果Lambda要表达的函数方案已经存在于某个方法的实现中,那么则可以通过双冒号来引用该方法作为Lambda的替代者。
2.通过5种语法使用方法引用
2.1 通过对象名引用成员方法

  //函数式接口    @FunctionalInterface    interface Printable{        public void print(String s);    }    //已存在的类,类中有打印字符串的方法.    class AlreadyExistPrint{        public void PrintString(String s){        System.out.println(s);    }    //测试通过对象名进行方法引用.    public class ObjectMethodReference {        public static void main(String[] args) {            //通过对象名引用方法            AlreadyExistPrint ap = new AlreadyExistPrint();            print(ap :: PrintString,"java");        }        public static void print(Printable p,String s){            p.print(s);        }    }    输出结果:            java

2.2 通过类名引用静态方法

  //函数式接口    @FunctionalInterface    interface MyMath{        int max(int a, int b);    }    //已存在的类    这里使用JDK提供的Math类中的静态方法max(int a,int b);    //测试通过类名引用静态方法.    public class ClassStaticMethod {        public static void main(String[] args) {            //通过Math类的类名引用静态方法max();            int max = getMax(Math :: max, 10, 20);            System.out.println(max);        }        public static int getMax(MyMath lambda ,int a,int b){            return lambda.max(a, b);        }    }    输出结果:            20

2.3 通过类名引用成员方法

/*    成员方法需要依托对象才可以执行,所以当并不存在对象时,成员方法无法执行。    如果希望成员方法的引用中仅出现类名称而不出现对象名称,情况则要复杂一些:    必须为其指定一个用来执行成员方法的对象实例。    */    //在函数式接口中的参数里加入对象实例:    @FunctionalInterface    interface Printable{        public void print(AlreadyExistPrint a ,String s);    }    //已存在的类    class AlreadyExistPrint{        public void PrintString(String s){            System.out.println(s);        }    }    //测试通过类名引用成员方法.    public class ClassMethod {        public static void main(String[] args) {            //通过类名引用成员方法;            //lambda方式实现:            //语义解析: 拿着对象a去调用a的print方法打印s             print((a,s) -> {a.PrintString(s);},new AlreadyExistPrint(),"hello");            //简化写法: 方法引用 (通过类名引用成员方法)            print(AlreadyExistPrint :: PrintString,new AlreadyExistPrint(),"hello");        }        public static void print(Printable p ,AlreadyExistPrint a,String s){            p.print(a,s);        }    }    输出结果:            hello            hello

2.4 通过super引用成员方法

    //函数式接口    @FunctionalInterface    interface Eat{        void eat(String food);    }    //已存在的类    class Animal{        public void eat(String food){            System.out.println("吃:"+food);        }    }    //Animal的子类    class Cat extends Animal{        @Override        public void eat(String food) {            //通过super引用父类的方法.            method(super :: eat,food);        }        public void method(Eat e, String s){            e.eat(s);        }    }    //测试通过类名引用成员方法.    public class SuperMethod {        public static void main(String[] args) {            Cat c = new Cat();            c.eat("鱼");        }    }    输出结果:            吃鱼

2.5通过this引用成员方法

    //函数式接口    interface shopping {        void buy(int money);    }    //已存在的类    class Man {        //男人将来都要买个属于自己的房子.        public void buyHouse(int money) {            System.out.println("买套房子消费:"+money);        }        //结婚就得购物,买买买啊.        public void marry(shopping lambda,int money) {            lambda.buy(money);        }        //开心方法.男人要想开心.就得结婚        public void beHappy() {            //通过this引用成员方法.            marry(this::buyHouse,1000000);        }    }    //测试通过类名引用成员方法.    public class ThisMethod {        public static void main(String[] args) {            Man man = new Man();            man.beHappy();        }    }    输出结果:            买套房子消费:1000000

3.方法引用使用总结
       (1) Lambda表达式:s -> System.out.println(s);        (2) 方法引用:System.out::println

第一种语义是指:拿到参数之后经Lambda之手,继而传递给System.out.println方法去处理。

第二种等效写法的语义是指:直接让System.out中的println方法来取代Lambda。
两种写法的执行效果完全一样,而第二种方法引用的写法复用了已有方案,更加简洁。

函数式接口是Lambda的基础,而方法引用是Lambda的孪生兄弟。

点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消