Heim >Java >javaLernprogramm >Eine eingehende Analyse von Methodenreferenzen in Java
Dieser Artikel vermittelt Ihnen relevantes Wissen über Java, in dem hauptsächlich verwandte Themen zur Methodenreferenz vorgestellt werden. Jeder kennt die Methode, die wir beim Schreiben von Code definieren. Die Methodenreferenz wird verwendet, um auf diese Methode zu verweisen. Die Referenzmethode macht deutlich, dass ihr Zweck darin besteht, Lambda-Ausdrücke weiter zu optimieren und dadurch den zu schreibenden Code einfacher zu machen. Ich hoffe, dass sie für alle hilfreich ist.
Empfohlene Studie: „Java-Video-Tutorial“
Tatsächlich beginnt jeder die Methode zu verstehen, die wir beim Schreiben von Code definieren . Die Methodenreferenz wird verwendet, um auf diese Methode zu verweisen. Die Referenzmethode macht deutlich, dass ihr Zweck darin besteht, Lambda-Ausdrücke weiter zu optimieren und dadurch den zu schreibenden Code einfacher zu machen. Rechts! Sie haben richtig gehört, Lambda-Ausdrücke sind bereits sehr optimiert. Wie können sie also optimiert werden? Wenn die entsprechende Klasse, das entsprechende Objekt, Super und This in unserem Code erscheinen, können wir Methodenreferenzen verwenden. Die Voraussetzung dieser Methodenreferenz ist, dass wir Lambda-Ausdrücke haben. Wie wird es also verwendet? Schauen wir weiter nach unten.
Da der Titel ein Methodenzitat ist, was ist ein Methodenzitat? Der Methodenreferenzoperator ist der Doppelpunkt [::], der die Methodenreferenz ist, und dies ist auch eine neue Syntax, die ein Referenzoperator ist, und die Methodenreferenz wird dadurch implementiert. Wenn die durch Lambda auszudrückende Funktionslösung bereits in der Implementierung einer Methode vorhanden ist, können wir einen Doppelpunkt verwenden, um auf die Methode zu verweisen und Lambda zu ersetzen.
Hinweis: Die in Lambda übergebenen Parameter müssen von einem für die Methode in der Methodenreferenz akzeptablen Typ sein, andernfalls wird eine Ausnahme ausgelöst.
Methodenreferenzen können auf folgende Weise verwendet werden:
Da es Methodenreferenzen auf die oben genannten Arten gibt, untersuchen wir sie einzeln.
Wir wissen, dass Objekte durch Klassen erstellt werden. Daher müssen wir zuerst eine Klasse erstellen, dann eine Mitgliedsmethode in der Klasse definieren, dann ein Objekt über die Klasse erstellen und Paare verwenden, um auf die Mitgliedsmethode zu verweisen.
Zum Beispiel:
Definieren Sie eine Mitgliedsmethode, übergeben Sie eine Zeichenfolge und geben Sie die Zeichenfolge in Großbuchstaben aus.
Lassen Sie uns die oben genannten Anforderungen implementieren.
Definieren Sie zuerst eine Klasse
public class Demo02MethodRerObject { //定义一个成员方法,传递字符串,把字符串按照大写输出 public void printUpperCaseString(String s){ System.out.println(s.toUpperCase()); } }
Da es sich um eine Ausgabe handelt, müssen wir sie ausdrucken. Für die Verwendung von Lambdab müssen wir eine funktionale Schnittstelle zum Drucken und eine abstrakte Methode zum Drucken von Zeichenfolgen in der funktionalen Schnittstelle definieren.
/* 定义一个打印的函数式接口 */ @FunctionalInterface public interface Printable { //定义打印字符串的抽象方法 void print(String s); }
Die Prämisse der Verwendung des Objektnamens zur Verweise auf die Mitgliedsmethode besteht darin, dass der Objektname bereits vorhanden ist und die Mitgliedsmethode auch vorhanden ist, können Sie den Objektnamen verwenden, um sich auf die Mitgliedsmethode zu beziehen. Schreiben wir es in Code: Zuerst verwenden wir Lambda, um diese Anforderung zu schreiben, und dann verwenden wir Methodenreferenzen, um Lambda zu optimieren.
public class Demo03ObjectMethodReference { //定义一个方法,方法参数传递Printable接口 public static void pringString(Printable p){ p.print("abcde"); } public static void main(String[] args) { //pringString(System.out::print); //调用printString方法,方法的参数pringable是一个函数式接口,所以可以传递Lambda pringString((s)->{ //创建MethodRerObject对象 Demo02MethodRerObject methodRerObject=new Demo02MethodRerObject(); //调用Demo02MethodRerObject对象中的成员方法printUpperCaseString,把字符串按照大写输出 methodRerObject.printUpperCaseString(s); }); /* 使用方法引用优化Lambda 对象已经存在Demo02MethodRerObject 成员方法也是已经存在的printUpperCaseString 所以我们可以使用对象名引用成员方法 */ Demo02MethodRerObject methodRerObject=new Demo02MethodRerObject(); pringString(methodRerObject::printUpperCaseString); } }
Wir haben zuvor gelernt, dass wir statische Methoden über den Klassennamen aufrufen können, wenn es statische Methoden in unserer Klasse gibt, und das Gleiche gilt auch für Methodenreferenzen Verweisen Sie auf statische Methoden über den Klassennamen. Im Folgenden verwenden wir auch Code zur Demonstration.
Dieses Mal definieren wir eine Methode, Die Parameter der Methode werden übergeben, um den absoluten Wert der Ganzzahl und der Funktionsschnittstelle Calcable zu berechnen.
Zuerst eine Schnittstelle definieren
@FunctionalInterface public interface Calcable { //定义一个抽象方法,传递一个整数,对整数进行绝对值计算并返回 int AbsCals(int number); }
wird über den Klassennamen referenziert, um auf die statische Member-Methode zu verweisen. Außerdem erstellen wir zuerst die Klasse, definieren die Methode, schreiben den Code mit Lambda und verwenden dann die Methodenreferenzoptimierung.
public class Demo04StaticMethodReference { //定义一个方法,方法的参数传递计算绝对值的整数和函数式接口Calcable public static int method1(int number,Calcable c){ return c.AbsCals(number); } public static void main(String[] args) { //调用method方法,传递计算绝对值的整数和lambda表达式 int number=method1(-10,(n)->{ //对参数进行绝对值计算并返回结果 return Math.abs(n); }); System.out.println(number); /* 使用方法引用优化Lambdab表达式 Math类是存在的 abs计算绝对值的静态方法也是存在的 所以我们可以直接通过类名引用静态方法 */ int number2=method1(-10, Math::abs); System.out.println(number2); } }u über Super, um die Mitgliedsmethode zu zitieren
erwähnte, dass die Super-Erklärung mit der übergeordneten Methode, also der Vererbung, zusammenhängt. Wenn eine Vererbungsbeziehung besteht und super in Lambda aufgerufen wird, haben wir stattdessen eine Methodenreferenz.
Definieren Sie eine Besprechungsmethode
Wir verwenden die Methode der Besprechung und Begrüßung zwischen Kinder- und Elternklassen zur DemonstrationAuch dieses Mal definieren wir die funktionale Schnittstelle der Besprechung/* 定义见面的函数式接口 */ @FunctionalInterface public interface Greetable { //定义一个见面的方法 void greet(); }Da wir erben müssen, definieren wir a Elternklasse
/* 定义父类方法 */ public class Demo05Fu_Human { //定义一个sayHello的方法 public void sayHello(){ System.out.println("Hello! 我是Human。"); } }
再定义一个子类,在子类中出现父类的成员方法,先使用Lambda编写代码,再进行方法引用优化。
使用super引用父类的成员方法,前提super是已经存在的,父类的成员方法也是存在的,就可以直接使用super引用父类成员方法。
import java.nio.channels.ShutdownChannelGroupException; /* 定义子类 */ public class Demo06Zi_Man extends Demo05Fu_Human { //子类重写父类sayHello方法 @Override public void sayHello() { System.out.println("Hello!我是Man。"); } //定义一个方法,参数传递Gerrtable接口 public void method(Greetable g){ g.greet(); } public void show(){ //调用method方法,方法参数Greetable是一个函数式接口,所以可以传递Lambda表达式 method(()->{ //创建父类的Human对象 Demo05Fu_Human fHuman=new Demo05Fu_Human(); fHuman.sayHello(); }); //因为有子父类关系,所以存在的一个关键super,代表父类,可以直接使用super调用父类的成员方法 method(()->{ super.sayHello(); }); /* 使用super引用类的成员方法 super是已经存在的 父类的成员方法也是存在的 使用可以直接使用super引用父类成员方法 */ method(super::sayHello); } public static void main(String[] args) { //调用show方法 new Demo06Zi_Man().show(); } }
既然上面用super引用了父类的成员方法,我们之前也学过this也可以调用本类的成员方法,那同样this也可以引用本类的成员方法。
示例:
定义一个买房子的方法
同样,首先定义函数式接口。
/* 定义一个富有的函数式接口 */ @FunctionalInterface public interface Richable { //定义一个想买什么就买什么的方法 void buy(); }
然后怎么创建类,再定义买房子的方法。通过this引用成员方法,前提this是已经存在的,买房子的成员方法也是存在的,就可以直接使用this引用成员方法。同样先使用Lambda编写代码,再进行方法引用优化。
/* 通过this引用本类的成员方法 */ public class Demo07_Husband { //定义一个买房子的方法 public void buyHouse(){ System.out.println("北京二环内买一套四合院!"); } //定义一个结婚的方法,参数传递Richable接口 public void marry(Richable r){ r.buy(); } //定义一个高兴的方法 public void soHappy(){ //调用结婚的方法,方法的参数Richable是一个函数式接口,传递Lambda表达式 marry(()->{ //使用this,成员方法,调用本类买房子的方法 this.buyHouse(); }); /* 使用方法引用优化Lambda this是已经存在的 买房子的成员方法也是存在的 可以直接使用this引用成员方法 */ marry(this::buyHouse); } public static void main(String[] args) { new Demo07_Husband().soHappy(); } }
类的构造器引用也叫构造方法引用。而由于构造器名称和类名完全一样,所以构造器引用格式是这样的,类名称::new的格式表示。既然是构造器引用也就是构造方法引用,所以我们需要:
定义一个Person类。
/* person类 */ public class Person { private String name; public Person() { super(); // TODO Auto-generated constructor stub } public Person(String name) { super(); this.name = name; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
然后创建一个Person对象的函数式接口
* 定义一个创建erson对象的函数式接口 */ @FunctionalInterface public interface PersonBuilder { //定义一个方法,根据传递的姓名,创建person对象返回 Person buliderPerson(String name); }
再传递一个方法,参数传递姓名和PersonBulider接口,方法中通过 姓名创建Person对象。类的构造器引用,前提构造方法new Person(String name)已知,创建对象已知 new,就可以使用Person引用new创建对象。同样先使用Lambda编写代码,再进行方法引用优化。
/* 类的构造器(构造方法)引用 */ import java.time.chrono.MinguoChronology; import javax.print.attribute.standard.PrinterName; public class Demo08Person { //传递一个方法,参数传递姓名和PersonBulider接口,方法中通过 姓名创建Person对象 public static void printName(String name,PersonBuilder pb){ Person person=pb.buliderPerson(name); System.out.println(person.getName()); } public static void main(String[] args) { //调用printName方法,方法的参数传递了函数式接口,我们可以使用Lambda表达式 printName("张三",(name)->{ return new Person(name); }); /*使用方法引用优化Lambda表达式 构造方法new Person(String name)已知 创建对象已知 new 就可以使用Person引用new创建对象*/ printName("痛而不言笑而不语的浅伤",Person::new); } }
数组也是Object的子类,所以它也有方法引用,只是语法上稍有不同。
示例:
定义一个方法
方法的参数传递创建数组的长度和ArrayBulider接口
方法内部根据创建的长度使用ArrayBuilder中的方法创建数组并返回
同样,先创建一个数组的函数式接口
/* 定义一个创建数组的函数式接口 */ @FunctionalInterface public interface ArrayBulider { // 定义一个int类型的数组方法,参数传递数组的长度,返回创建好的int类型的数组 int[] buliderArray(int length); }
方法的参数传递创建数组的长度和ArrayBulider接口,方法内部根据创建的长度使用ArrayBuilder中的方法创建数组并返回。前提,已知创建的就是int[]数组,数组的长度也是已知的,就可以通过数组int[]引用new,根据参数传递的长度来创建数组同样先使用Lambda编写代码,再进行方法引用优化。
import java.lang.reflect.Array; import java.util.Arrays; /* 数组的构造器引用 */ public class Demo09Array_BuilderArray { /* 定义一个方法 方法的参数传递创建数组的长度和ArrayBulider接口 方法内部根据创建的长度使用ArrayBuilder中的方法创建数组并返回 */ public static int[] arrayLength(int length,ArrayBulider ab){ return ab.buliderArray(length); } public static void main(String[] args) { //调用arrayLength方法、传递数组的长度和Lambda表达式 int[]arr=arrayLength(10,(len)->{ return new int[len]; }); System.out.println(arr.length); /*使用方法引用优化Lambda表达式 已知创建的就是int[]数组 数组的长度也是已知的 就可以通过数组int[]引用new,根据参数传递的长度来创建数组*/ int[]arr1=arrayLength(5, int[]::new); System.out.println(arr1.length); System.out.println(Arrays.toString(arr1)); } }
推荐学习:《java视频教程》
Das obige ist der detaillierte Inhalt vonEine eingehende Analyse von Methodenreferenzen in Java. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!