怎么理解Java里的双冒号 “::”

 “::”是什么?

在java中,双冒号“::”是方法引用的语法。方法引用是简化Lambda表达式的语法结构,使代码更简洁易懂,并且在使用引用方法时,会根据上下文推断参数类型,因此特别适用于直接引用已有方法的情况。

“::”的用法

 1 ClassName::MethodName 

其实,Class是包含静态方法methodName的类名,根据引用的方法类型不同,有不同的情况:

1.静态方法引用

假设我们有一个自定义的工具类MathUtil,其中就包含了一个静态方法square,用于计算一个整数的平方。现在我们想要计算一个整数列表中所有元素的平方和。

 1 import java.util.Arrays;
 2 import java.util.List;
 3 
 4 public class MathUtil {
 5 
 6 
 7     public static int square(int num){
 8         System.out.println(num * num);
 9         return num * num;
10     }
11 
12     public static int square(){
13         System.out.println(1);
14         return 1;
15     }
16 
17 
18     public static void main(String[] args) {
19         List<Integer> numbers = Arrays.asList(1,2,3,4,5);
20         int sum = numbers.stream().mapToInt(MathUtil::square).sum();
21         System.out.println(sum);
22     }
23 }

在上述方法中,我们通过使用静态方法引用MathUtil::square,将数组中的数组转成int类型,再使用mapToInt方法循环调用square(int num)方法,以便对列表中每个元素进行平方运算。

 

2.实例方法的引用

假设我们有一个字符串列表,我们想要按照字符串长度进行排序,我们可以使用Lambda表达式编写比较器,也可以使用实例方法引用简化代码。

 1 import java.util.Arrays;
 2 import java.util.List;
 3 
 4 public class StringUtil {
 5     public static void main(String[] args) {
 6         List<String> phoneBrand = Arrays.asList("apple", "xiaomi", "huawei", "vivo", "oppo", "meizu");
 7         phoneBrand.sort((a,b) -> a.compareTo(b));
 8         System.out.println(phoneBrand); //[apple, huawei, meizu, oppo, vivo, xiaomi]
 9 
10         //实例方法引入
11         phoneBrand.sort(String::compareTo);
12         System.out.println(phoneBrand); //[apple, huawei, meizu, oppo, vivo, xiaomi]
13     }
14 }

在上述代码总,我们首先使用了Lambda编写了一个比较器进行字符串比较,然后又使用实例方法引用来简化比较器的写法,无需传递参数,直接调用方法,

 

3.对象方法引用

假设我们有一个自定义的Student类,其中包含姓名和年龄属性。我们想要根据student对象的年龄进行排序。

 1 import com.alibaba.fastjson.JSON;
 2 import java.util.Arrays;
 3 import java.util.List;
 4 
 5 public class ObjectUtil {
 6 
 7 
 8     public static void main(String[] args) {
 9         //示例1
10         List<Student> students = Arrays.asList(
11                 new Student("小刘",18),
12                 new Student("小张",20),
13                 new Student("小王",22)
14         );
15 
16         //使用Lambda编写比较器
17         students.sort((s1,s2) -> s1.getAge() - s2.getAge());
18         System.out.println(JSON.toJSONString(students));    //[{"age":18},{"age":20},{"age":22}]
19 
20         //使用对象方法引用
21         students.sort(Student::compareAge);
22         System.out.println(JSON.toJSONString(students));    //[{"age":18},{"age":20},{"age":22}]
23 
24 
25 
26 
27 
28         //示例2
29         List<Integer> numbers = Arrays.asList(1,2,3,4,5);
30         numbers.sort(ObjectUtil::sum);  //3----5----7----9----
31 
32     }
33 
34     //示例2
35     public static int sum(int num1,int num2){
36         System.out.print(num1 + num2 + "----");
37         return num1 + num2;
38     }
39 
40     //示例1
41     static class Student {
42 
43         private String name;
44         private int age;
45 
46         public Student(String name, int age) {
47             this.name = name;
48             this.age = age;
49         }
50 
51         public int getAge() {
52             return age;
53         }
54 
55         public void setAge(int age) {
56             this.age = age;
57         }
58 
59         public static int compareAge(Student s1, Student s2){
60             return s1.getAge() - s2.getAge();
61         }
62     }
63 }

上述代码中,我们首先使用Lambda编写了一个比较器来根据年龄进行排序。然后我们又使用对象方法引用Student::compareAge来简化比较器的写法。其实个人觉得和实例方法引用没什么区别,无非就是一个引用自定义的,一个引用别人写好的。

4.构造方法引用

假设我们需要创建一个空的ArrayList,可以使用构造方法引用来实现。

 1 import com.google.common.base.Supplier;
 2 import java.util.ArrayList;
 3 import java.util.List;
 4 
 5 public class StructureUtil {
 6     public static void main(String[] args) {
 7         Supplier<List<String>> supplier = ArrayList::new;
 8         List<String> strings = supplier.get();
 9         System.out.println(strings instanceof List);    //true
10     }
11 }
posted @ 2024-06-11 23:40  刘小铭  阅读(267)  评论(0编辑  收藏  举报