泛型的通配符

这篇具有很好参考价值的文章主要介绍了泛型的通配符。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

泛型的通配符

一、泛型的上界

class Myclass<类型参数 extends 类型上界> {

}

类型的上界决定了泛型的范围。

class MyArray<T extends Number> {
    
}
public static void main(String[] args) {
        MyArray<Integer> myArray = new MyArray<>();
        MyArray<Boolean> myArray1 = new MyArray<Boolean>();
    }

泛型的通配符
我们发现指定了泛型的上界为数值类Number时,传入Boolean类型就会报错。
如果没有指定类型的边界,可以认可 T extends Object,当指定了某个类型为上界,那么只接受某类型本身和子类型作为E的类型实参

上界为接口

我们要实现一个类去找数组的一个的最大值
泛型的通配符
这里报错了,因为我们这里是Object类型,如果是类型实参是对象的话不就能这样比较了,所以我们将上界定为Comparable接口。

class MyArray<T extends Comparable<T>> {
   
    public T findMax(T[] arr) {
        T max = arr[0];
        for (int i = 1; i < arr.length - 1; i++) {
            if (arr[i].compareTo(max) > 0) {
                max = arr[i];
            }
        }
        return max;
    }
}

我们来求一个Integer类型的最大值。

public static void main(String[] args) {
        MyArray<Integer> myArray = new MyArray<>();
        Integer[] arr = {1,4,6,3,2};
        System.out.println(myArray.findMax(arr));
    }

泛型的通配符
我们来求一个Person类的最大值。

泛型的通配符
我们发现使用Person作为类型实参,未法创建对象,因为我们将泛型上界设置为了实现了Comparable的类类型。

class Person implements Comparable<Person>{
    public int age;

    public Person(int age) {
        this.age = age;
    }

    @Override
    public int compareTo(Person o) {
        return this.age - o.age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "age=" + age +
                '}';
    }
}
public static void main(String[] args) {
        MyArray<Person> myArray = new MyArray<>();
        Person[] per = {new Person(15),new Person(18),new Person(20)};
        System.out.println(myArray.findMax(per));
    }

泛型的通配符

二、泛型方法

1 静态方法

刚刚我们实现的求某个类型数组的最大值必须得创建一个对象,那我们可以不创建对象直接调用吗?
泛型的通配符
我们直接给方法加上static时,系统报错了,因为static方法是在类加载指定,擦除机制是在编译时检查和强制转换,在类加载时,并不知道是什么类型,所以不能直接加static。
静态方法语法:

public static<泛型形参> T findMax(T[] arr)
class MyArray {
    public static<T extends Comparable<T>>  T findMax(T[] arr) {
        T max = arr[0];
        for (int i = 1; i < arr.length; i++) {
            if (arr[i].compareTo(max) > 0) {
                max = arr[i];
            }
        }
        return max;
    }
}
public static void main(String[] args) {
        Integer[] arr = {1,5,4,7,9,2};
        Integer max = MyArray.findMax(arr);
        System.out.println(max);
    }

泛型的通配符
这样我们就可以不依赖对象求出最大值了,那类型实参在那里传?
泛型的通配符
一般在引用点的后面,一般可以省略。

2 普通方法

静态方法语法:

public <泛型形参> T findMax(T[] arr)
class MyArray {
    public <T extends Comparable<T>>  T findMax(T[] arr) {
        T max = arr[0];
        for (int i = 1; i < arr.length; i++) {
            if (arr[i].compareTo(max) > 0) {
                max = arr[i];
            }
        }
        return max;
    }
}
public static void main(String[] args) {
        Integer[] arr = {1,5,4,7,9,2};
        MyArray myArray = new MyArray();
        System.out.println(myArray.findMax(arr));
    }

泛型的通配符

三、通配符

什么是通配符:? 用于在泛型的使用

1 通配符的引入

class Data<T> {
    public T data;

    public T getData() {
        return data;
    }

    public void setData(T data) {
        this.data = data;
    }
}
public static void main(String[] args) {
        Data<Integer> data = new Data<>();
        data.setData(18);
        print(data);
    }
    public static void print(Data<Integer> data) {
        System.out.println(data.getData());
    }

泛型的通配符
如果我们想输入类型实参为String类型呢?重写一个方法吗?支持重载吗?
泛型的通配符
我们当写入两种不同泛型类型的方法时,系统报错了,因为在运行阶段是没有泛型这个概念的,相当于他们的类型都是Data类型。
我们来验证一下,打印两种不同泛型实参类型的对象:

public static void main(String[] args) {
        Data<Integer> data = new Data<>();
        Data<String> data1 = new Data<>();
        System.out.println(data);
        System.out.println(data1);
    }

泛型的通配符
我们发现打印出来对象的类型是一样的。

2 通配符的使用

我们需要的解决方案:可以接收所有的泛型类型,但是又不能够让用户随意修改。这种情况就需要使用通配符"?"来处理

public static void main(String[] args) {
        Data<Integer> data = new Data<>();
        data.setData(10);
        print(data);
        Data<String> data1 = new Data<>();
        data1.setData("woyaojindachang");
        print(data1);
    }
    //此处?代表可以接受任意类型,但由于类型不确定,所以无法修改
    public static void print(Data<?> data) {
        System.out.println(data.getData());
    }

泛型的通配符

3 通配符的上界

? extends 类:设置泛型上限

class Data<T> {
    public T data;
    
    public T getData() {
        return this.data;
    }
    
    public void setData(T data) {
        this.data = data;
    }
}
public class Test {
    public static void main(String[] args) {
        Data<Integer> data = new Data<>();
        data.setData(10);
        print(data);
        Data<String> data1 = new Data<>();
        data1.setData("woyaojindachang");
        print(data1);//String不属于Number的子类
    }
    public static void print(Data<? extends Number> data) {
        System.out.println(data.getData());
    }
}

泛型的通配符
这里我们设置通配符的上界为Number,所以无法传入String。

泛型的通配符
因为此时通配符里可以接受任意类型,由于类型不确定,所以无法修改。

public static void print(Data<? extends Number> data) {
        Number number = data.getData();
    }

因为data存储的是Number和它的子类,所以可以进行读取数据。

4 通配符的下界

? super 类:设置泛型下限

public static void fun(Data<? super Integer> data) {
        data.setData(10);
    }

可以传入的实参的类型是Integer或者Integer的父类类型,此处可以修改,因为添加的是Integer或者它的父类。
泛型的通配符
但不能接收,因为无法确定是那个父类。

四、拆箱和装箱

手动装拆箱:

public static void main(String[] args) {
        //装箱
        Integer a = Integer.valueOf(10);
        Integer b = new Integer(10);
        //拆箱
        int c = a.intValue();
    }

每次手动装拆箱太过于麻烦,系统提供自动装拆箱。

public static void main(String[] args) {
        //自动装箱
        Integer a = 10;
        Integer b = (Integer)10;
        //自动拆箱
        int j = a;
        int k = (int)a;
    }

泛型的通配符
我们打开字节码看一下,发现系统自动在编译阶段调用了Integer.valueof和Integer.intValue方法

public static void main(String[] args) {
        Integer a = 127;
        Integer b = 127;
        Integer c = 200;
        Integer d = 200;
        System.out.println(a == b);
        System.out.println(c == d);
    }

经典面试题

上面这段代码会输出什么呢?
泛型的通配符
泛型的通配符
我们可以发现当数值在一定的范围内时,返回的是固定引用,只有超出范围才会new 新对象。
泛型的通配符
泛型的通配符
我们可以得出 i 是介于-128 - 127之间的。
泛型的通配符
cache数组存储着这些引用,并且final不可修改。
Byte、Short、Integer、Long这四种包装类型默认创建了[-128,127]的相应类型缓存数据,Character创建了数值在[0,127]范围的缓存数据,Boolean直接返回true或者false。文章来源地址https://www.toymoban.com/news/detail-403305.html

到了这里,关于泛型的通配符的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 深入理解Java泛型、协变逆变、泛型通配符、自限定

    重写了之前博客写的泛型相关内容,全部整合到这一篇文章里了,把坑都填了,后续不再纠结这些问题了。本文深度总结了函数式思想、泛型对在Java中的应用,解答了许多比较难的问题。 纯函数 协变 逆变 泛型通配符 PECS法则 自限定 Java8 引入了函数式接口,从此方法传参可

    2023年04月20日
    浏览(37)
  • 【类型通配符】

    为了表示各种泛型List的父类,可以使用类型通配符 类型通配符:? List?:表示元素类型未知的List,它的元素可以匹配任何的类型 这种带通配符的List仅表示它是各种泛型List的父类,并不能把元素添加到其中 如果不想让List?是任何泛型的父类,只想让它代表某一类泛型List的父

    2024年02月17日
    浏览(38)
  • 详解Java中的泛型(泛型的语法,擦除机制,泛型的上界)

    目录 一.什么是泛型 二.Java中为什么要使用泛型 三.泛型的语法 四.泛型类的使用 五.泛型的编译机制(擦除机制) 六.泛型的上界 泛型(Generics)是Java SE 5中引入的一个新特性,可以 使Java中的类和方法具有更广泛的类型范围 。通俗的说,它使得我们可以在定义类和方法时指定

    2024年02月05日
    浏览(40)
  • 活用 命令行通配符

    本文是对 阮一峰老师 命令行通配符教程 [1] 的学习与记录 通配符早于正则表达式出现,可以看作是原始的正则表达式. 其功能没有正则那么强大灵活,而胜在简单和方便. - 字符 切回上一个路径/分支 如图: !! 代表上一个命令, 如图: [Linux中“!\\\"的神奇用法](https://www.cnblogs.com/bian

    2024年02月10日
    浏览(43)
  • Linux详解:通配符

    Linux是一款开源操作系统,其灵活性和可定制性一直受到开发者的喜爱和追捧。而且,Linux在文件管理方面提供了丰富的功能,例如通配符,它是一种用于匹配文件名的特殊字符。通配符在Linux中可以帮助我们更加方便和快捷地查找和操作文件。本文将介绍Linux中常用的通配符

    2024年02月09日
    浏览(47)
  • Elasticsearch 通配符查询

    通配符查询(wildcard query) 匹配字段被通配符表达式(没有被分析)匹配的文档。支持的通配符为*(匹配任意字符序列,包括空字符序列)以及?(匹配任意单字符)。注意,此查询可能会很慢,它需要迭代许多字段值。为了防止极慢的通配符匹配,通配符字段值不能以一个

    2024年02月11日
    浏览(70)
  • 【算法题】44. 通配符匹配

    给你一个输入字符串 (s) 和一个字符模式 (p) ,请你实现一个支持 \\\'?\\\' 和 \\\'*\\\' 匹配规则的通配符匹配: \\\'?\\\' 可以匹配任何单个字符。 \\\'*\\\' 可以匹配任意字符序列(包括空字符序列)。 判定匹配成功的充要条件是:字符模式必须能够 完全匹配 输入字符串(而不是部分匹配)。  

    2024年01月25日
    浏览(37)
  • 44. 通配符匹配(动态规划)

    Problem: 44. 通配符匹配 给你一个输入字符串 (s) 和一个字符模式p ,请你实现一个支持 ‘?’ 和 ‘ ’ 匹配规则的通配符匹配: ‘?’ 可以匹配任何单个字符。 \\\' ’ 可以匹配任意字符序列(包括空字符序列)。 判定匹配成功的充要条件是:字符模式必须能够 完全匹配 输入字符

    2024年02月04日
    浏览(40)
  • 16-字符串通配符

    题目 问题描述: 在计算机中,通配符一种特殊语法,广泛应用于文件搜索、数据库、正则表达式等领域。现要求各位实现字符串通配符的算法。 要求: 实现如下2个通配符: *:匹配0个或以上的字符(注:能被*和?匹配的字符仅由英文字母和数字0到9组成,下同) ?:匹配

    2024年02月15日
    浏览(32)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包