当前位置: 首页 > wzjs >正文

网站建设资料免费源码分享论坛

网站建设资料,免费源码分享论坛,什么是标记型网页制作工具,大丰做网站的公司在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。那么考虑一种情况:如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑?…

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

1.1 冗余的Lambda场景

来看一个简单的函数式接口以应用Lambda表达式:

@FunctionalInterface
public interface Printable {void print(String str);
}

Printable接口当中唯一的抽象方法print接收一个字符串参数,目的就是为了打印显示它。那么通过Lambda来使用它的代码很简单:

public class Demo01PrintSimple {private static void printString(Printable data){data.print("Hello, World!");}public static void main(String[] args) {printString(str -> System.out.println(str));}
}

在这里插入图片描述

其中printString方法只管调用Printable接口的print方法,而并不管print方法的具体实现逻辑会将字符串打印到什么地方去。而main方法通过Lambda表达式指定了函数式接口Printable的具体操作方案为:拿到String(类型可推导,所以可省略)数据后,在控制台中输出它

1.2 问题分析

这段代码的问题在于,对字符串进行控制台打印输出的操作方案,明明已经有了现成的实现,那就是System.out对象中的println(String)方法。既然Lambda希望做的事情就是调用println(String)方法,那何必自己手动调用呢?

1.3 用方法引用改进代码

能否省去Lambda的语法格式(尽管它已经相当简洁)呢?只要“引用”过去就好了:

public class Demo02PrintRef {private static void printString(Printable data){data.print("Hello, World!");}public static void main(String[] args) {printString(System.out::println);}
}

在这里插入图片描述

请注意其中的双冒号:: 写法,这被称为“方法引用”,而双冒号是一种新的语法。

1.4 方法引用符

双冒号::为引用运算符,而它所在的表达式被称为方法引用。如果Lambda要表达的函数方案已经存在于某个方法的实现中,那么则可以通过双冒号来引用该方法作为Lambda的替代者。

语义分析

例如上例中,System.out对象中有一个重载的println(String)方法恰好就是我们所需要的。那么对于printString方法的函数式接口参数,对比下面两种写法,完全等效:

  • Lambda表达式写法:str -> System.out.println(str);

  • 方法引用写法:System.out::println

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

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

注:Lambda中 传递的参数 一定是方法引用中 的那个方法可以接收的类型,否则会抛出异常

推导与省略

如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定的重载形式——它们都将被自动推导。而如果使用方法引用,也是同样可以根据上下文进行推导。

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

下面这段代码将会调用println方法的不同重载形式,将函数式接口改为int类型的参数:

@FunctionalInterface
public interface Printable {void print(String str);
}

由于上下文变了之后可以自动推导出唯一对应的匹配重载,所以方法引用没有任何变化:

public class Demo03PrintOverload {private static void printInteger(PrintableInteger data) {data.print(2048);}public static void main(String[] args) {printInteger(System.out::println);}
}

在这里插入图片描述

这次方法引用将会自动匹配到println(int)的重载形式。

1.5 通过对象名引用成员方法

这是最常见的一种用法,与上例相同。如果一个类中已经存在了一个成员方法:

public class MethodRefObject {public void printUpperCase(String str) {System.out.println(str.toUpperCase());}
}

函数式接口仍然定义为:

@FunctionalInterface
public interface Printable {void print(String str);
}

那么当需要使用这个printUpperCase成员方法来替代Printable接口的Lambda的时候,已经具有了MethodRefObject类的对象实例,则可以通过对象名引用成员方法,代码为:

public class Demo04MethodRef {private static void printString(Printable lambda) {lambda.print("Hello");}public static void main(String[] args) {MethodRefObject obj = new MethodRefObject();printString(obj::printUpperCase);}
}

在这里插入图片描述

1.6 通过类名称引用静态方法

由于在java.lang.Math类中已经存在了静态方法abs,所以当我们需要通过Lambda来调用该方法时,有两种写法。首先是函数式接口:

@FunctionalInterface
public interface Calcable {int calc(int num);
}

第一种写法是使用Lambda表达式:

public class Demo05Lambda {private static void method(int num, Calcable lambda) {System.out.println(lambda.calc(num));}public static void main(String[] args) {method(-99,num -> Math.abs(num));}
}

在这里插入图片描述

但是使用方法引用的更好写法是:

public class Demo06MethodRef {private static void method(int num, Calcable lambda) {System.out.println(lambda.calc(num));}public static void main(String[] args) {method(-99, Math::abs);}
}

在这里插入图片描述

在这个例子中,下面两种写法是等效的:

  • Lambda表达式:num -> Math.abs(num)

  • 方法引用: Math::abs

1.7 通过super引用成员方法

如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。首先是函数式接口:

@FunctionalInterface
public interface Greetable {void greet();
}

然后是父类Human的内容:

public class Human {public void sayHello() {System.out.println("Hello!");}
}

最后是子类 Man 的内容,其中使用了Lambda的写法:

public class Man extends Human{@Overridepublic void sayHello() {System.out.println("大家好,我是Man!");}//定义方法method,参数传递Greetable接口public void method(Greetable g){g.greet();}public void show(){//调用method方法,使用Lambda表达式method(()->{new Human().sayHello(); //创建Human对象,调用sayHello方法});//简化Lambdamethod(()->new Human().sayHello());//使用super关键字代替父类对象method(()->super.sayHello());}
}

但是如果使用方法引用来调用父类中的sayHello方法会更好,例如另一个子类Woman

public class Woman  extends Human{@Overridepublic void sayHello() {System.out.println("大家好,我是Woman!");}//定义方法method,参数传递Greetable接口public void method(Greetable g){g.greet();}public void show(){method(super::sayHello);}
}

在这个例子中,下面两种写法是等效的:

  • Lambda表达式:() -> super.sayHello()

  • 方法引用:super::sayHello

1.8 通过this引用成员方法

this代表当前对象,如果需要引用的方法就是当前类中的成员方法,那么可以使用“this::成员方法”的格式来使用方法引用。首先是简单的函数式接口:

@FunctionalInterface
public interface Richable {void buy();
}

下面是一个丈夫Husband类:

public class Husband {private void marry(Richable lambda) {lambda.buy();}public void beHappy() {marry(() -> System.out.println("买套房子"));}
}

开心方法beHappy调用了结婚方法marry,后者的参数为函数式接口Richable,所以需要一个Lambda表达式。但是如果这个Lambda表达式的内容已经在本类当中存在了,则可以对Husband丈夫类进行修改:

public class Husband {private void buyHouse() {System.out.println("买套房子");}private void marry(Richable lambda) {lambda.buy();}public void beHappy() {marry(() -> this.buyHouse());}
}

如果希望取消掉Lambda表达式,用方法引用进行替换,则更好的写法为:

public class Husband {private void buyHouse() {System.out.println("买套房子");}private void marry(Richable lambda) {lambda.buy();}public void beHappy() {//marry(() -> this.buyHouse());marry(this::buyHouse);}
}

在这个例子中,下面两种写法是等效的:

  • Lambda表达式:() -> this.buyHouse()

  • 方法引用:this::buyHouse

1.9 类的构造器引用

由于构造器的名称与类名完全一样,并不固定。所以构造器引用使用类名称::new的格式表示。首先是一个简单的Person类:

public class Person {private String name;public Person(String name) {this.name = name;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}

然后是用来创建Person对象的函数式接口:

public interface PersonBuilder {Person buildPerson(String name);
}

要使用这个函数式接口,可以通过Lambda表达式:

public class Demo09Lambda {public static void printName(String name, PersonBuilder builder) {System.out.println(builder.buildPerson(name).getName());}public static void main(String[] args) {printName("张三", name ->  new Person(name));}
}

在这里插入图片描述

但是通过构造器引用,有更好的写法:

public class Demo10ConstructorRef {public static void printName(String name, PersonBuilder builder) {System.out.println(builder.buildPerson(name).getName());}public static void main(String[] args) {printName("张三", Person::new);}
}

在这里插入图片描述

在这个例子中,下面两种写法是等效的:

  • Lambda表达式:name -> new Person(name)

  • 方法引用: Person::new

1.10 数组的构造器引用

数组也是Object的子类对象,所以同样具有构造器,只是语法稍有不同。如果对应到Lambda的使用场景中时,需要一个函数式接口:

@FunctionalInterface
public interface ArrayBuilder {int[] buildArray(int length);
}

在应用该接口的时候,可以通过Lambda表达式:

public class Demo11ArrayInitRef {private static int[] initArray(int length, ArrayBuilder builder) {return builder.buildArray(length);}public static void main(String[] args) {int[] array = initArray(10, length -> new int[length]);}
}

但是更好的写法是使用数组的构造器引用:

public class Demo12ArrayInitRef {private static int[] initArray(int length, ArrayBuilder builder) {return builder.buildArray(length);}public static void main(String[] args) {int[] array = initArray(10, int[]::new);}
}

在这个例子中,下面两种写法是等效的:

  • Lambda表达式:length -> new int[length]

  • 方法引用:int[]::new


文章转载自:

http://r78zTXwK.kxnxf.cn
http://dpgrivZk.kxnxf.cn
http://i4O0Xrz0.kxnxf.cn
http://ekz4OyJ6.kxnxf.cn
http://vmoyziUw.kxnxf.cn
http://a9wAcyou.kxnxf.cn
http://NhPVR0s5.kxnxf.cn
http://ulxzsiK7.kxnxf.cn
http://omiDIR1k.kxnxf.cn
http://6HOVyLDp.kxnxf.cn
http://N04wzSBd.kxnxf.cn
http://jokJGdZc.kxnxf.cn
http://NzM2Lhio.kxnxf.cn
http://DgxIkeyt.kxnxf.cn
http://acn9iVqC.kxnxf.cn
http://CFxWqoft.kxnxf.cn
http://LULLi0Ds.kxnxf.cn
http://n23fWNkv.kxnxf.cn
http://3IJhmnxL.kxnxf.cn
http://lzEbthj7.kxnxf.cn
http://BjuRTTAA.kxnxf.cn
http://lceD3StV.kxnxf.cn
http://Qp3CGBMR.kxnxf.cn
http://QICZdv7t.kxnxf.cn
http://XZAAuH8j.kxnxf.cn
http://4q6QG4ke.kxnxf.cn
http://dcnnRQVI.kxnxf.cn
http://GbM12Uxd.kxnxf.cn
http://cOUdFp9o.kxnxf.cn
http://VFl222IJ.kxnxf.cn
http://www.dtcms.com/wzjs/656490.html

相关文章:

  • 网站建设方案格式为什么企业网站不是开源系统
  • 服饰网站建设模板ftp中打开wordpress
  • 免费搭建网站的软件seo搜索引擎优化工资
  • 营销型网站制作msgg简单企业网站源码 asp.net 公司介绍 产品展示
  • 制作婚恋网站做网站要考虑的
  • 网站建设工作室简介免费注册帐号qq
  • 网页设计与网站建设 公开课营销管理软件
  • 设计师常看的网站互联网公司运营
  • 昆明有多少做网站的公司网站如何做微信推广方案设计
  • 垂直网站建设方案在线平面广告设计
  • wordpress post in长沙优化排名推广
  • 170个可带链接锚文本外链的网站论坛事件营销ppt
  • 怎么在网站上做图片轮播网站开发语言 .net
  • 做一个网站做少多少钱上海行业网站建设
  • 免费网站建设排行榜开发区网站制作公司
  • 商丘网站建设网站后台添加新闻
  • 房产网站关键词优化建设网站的公司兴田德润实力强
  • 网站空间已过期创建全国文明城市的主体是什么
  • 网站建设安全服务协议做行业网站如何采集信息
  • 哪个着陆页网站企业注册查询网
  • 网站建设的目标的意思wordpress 增加备案
  • 学网站建设课程哪哪个网站可以做兼职
  • 游戏网站设计书全球搜效果怎么样
  • 惠州高端模板建站顺企网萍乡网站建设
  • 如何制作大气网站wordpress ie兼容插件
  • 城市建设与管理局网站做网站的最终目的
  • 南沙门户网站建设优帮云查询数据云查询
  • 个人免费网站建设模板官网是什么意思
  • 做家装壁纸的网站申请网站空间有哪几种方式
  • 基本信息型营销网站有哪些php网站制作教程