泛型通配符

泛型&通配符

文章目录

    • 泛型&通配符
    • 一、泛型介绍&理解
      • 1.1 泛型概述&使用(集合/比较器)
      • 1.2 自定义范型结构(类/接口/方法)
    • 二、通配符&读写特点
    • 三、企业真题


一、泛型介绍&理解

1.1 泛型概述&使用(集合/比较器)

泛型:类似于场景中的标签,即为类型参数,在声明它的类、接口或方法中,代表未知的某种通用类型。
比如使用集合存储数据时,除了元素的类型不确定,其他部分是确定的(例如关于这个元素如何保存,如何管理等)。
JDK1.5设计了泛型的概念,例如:JDK5.0改写了集合框架中的全部接口和类、java.lang.Comparable接口、java.util.Comparator接口、Class类等。为这些接口、类增加了泛型支持,从而可以在声明变量、创建对象时传入类型实参。

简单示例:不确定的类型用T来表示,如果不指定的话,可传入任意类型的

public class Teacher<T> {
    T name;
    int add(T num){
        return 0;
    }
}
//测试:不指定具体类型,缺点:会出现类型转换异常等等
@Test
public void test1(){
     Teacher tea = new Teacher();
     tea.name = 1;//可以是任意类型的
     tea.name ="23";
 }
 //测试:指定具体类型,此时所有T代表的类型参数全部替换为类型String
@Test
public void test2(){
     Teacher<String> tea = new Teacher<>();
     //tea.name = 1;//报错
     tea.name ="23";
 }

1.什么是泛型?
即类型参数,允许在定义类、接口、方法时通过一个标识表示类中某个属性的类型、某个方法的返回值或参数的类型
这个类型参数将在使用时(例如,继承或实现这个接口、创建对象或调用方法时)确定(即传入实际的类型参数,也称为类型实参)。

2、使用泛型之前可能存在的问题

  • 在集合中
    • 问题1:类型不安全,因为add()的参数是0bject类型,意味着任何类型的对象都可以添加成功
    • 问题2:需要使用强转操作,繁琐。还有可能导致ClassCastException异常。
  • 在比较器中
    • 不能确定是什么类型的对象比较大小,需要判断是否是该类型的实例、类型转换等操作,用了泛型之后,他就只能接受指定类型的对象

3、使用泛型的好处:以集合为例:把一个集合中的内容限制为一个特定的数据类型,这就是generic背后的核心思想。
Java泛型可以保证如果程序在编译时没有发出警告,运行时就不会产生ClassCastException异常。即,把不安全的因素在编译期间就排除了,而不是运行期;既然通过了编译,那么类型一定是符合要求的,就避免了类型转换。
比较器的泛型实现示例
集合使用泛型前后的对比

集合使用泛型示例:ArrayList,注意可以用类型推断
实现功能:使用集合的removeIf方法删除偶数,为Predicate接口指定泛型<Ineteger>

public class TestNumber {
    public static void main(String[] args) {
        ArrayList<Integer> coll = new ArrayList<Integer>();
        Random random = new Random();
        for (int i = 1; i <= 5 ; i++) {
            coll.add(random.nextInt(100));
        }
        System.out.println("coll中5个随机数是:");
        for (Integer integer : coll) {
            System.out.println(integer);
        }
        //方式1:使用集合的removeIf方法删除偶数
        coll.removeIf(new Predicate<Integer>() {
            @Override
            public boolean test(Integer integer) {
                return integer % 2 == 0;
            }
        });
        //方式2:调用Iterator接口的remove()方法
        //Iterator<Integer> iterator1 = coll.iterator();
        //while(coll.hasNext()){
        //    Integer i = coll.next();
        //   if(i % 2 == 0){
        //       coll.remove();
        //    }
        //}

        System.out.println("coll中删除偶数后:");
        Iterator<Integer> iterator = coll.iterator();
        while(iterator.hasNext()){
            Integer number = iterator.next();
            System.out.println(number);
        }
    }
}

集合使用泛型示例:HashMap,注意可以用类型推断

//泛型在Map中的使用
@Test
public void test2(){
	//HashMap<String,Integer> map = new HashMap<String,Integer>();
    HashMap<String,Integer> map = new HashMap<>();//使用类型推断
    map.put("Tom",67);
    map.put("Jim",56);
    map.put("Rose",88);
    //编译不通过
    //ap.put(67,"Jack");

    //遍历key集
    Set<String> keySet = map.keySet();
    for(String str:keySet){
        System.out.println(str);
    }

    //遍历value集
    //var values = map.values();//可以使用类型推断
    Collection<Integer> values = map.values();
    Iterator<Integer> iterator = values.iterator();
    while(iterator.hasNext()){
        Integer value = iterator.next();
        System.out.println(value);
    }
    //遍历entry集
    Set<Map.Entry<String, Integer>> entrySet = map.entrySet();
    Iterator<Map.Entry<String, Integer>> iterator1 = entrySet.iterator();
    while(iterator1.hasNext()){
        Map.Entry<String, Integer> entry = iterator1.next();
        String key = entry.getKey();
        Integer value = entry.getValue();
        System.out.println(key + ":" + value);
    }
}

Comparable接口使用泛型后

class CircleComparator1 implements Comparator<Circle> {//指定具体的泛型类型
    @Override
    public int compare(Circle o1, Circle o2) {
        //不再需要强制类型转换,代码更简洁
        return Double.compare(o1.getRadius(), o2.getRadius());
    }
}
//测试类
public class TestHasGeneric {
    public static void main(String[] args) {
        CircleComparator1 com = new CircleComparator1();
        System.out.println(com.compare(new Circle(1), new Circle(2)));//如果有错误,会在编译阶段就报错
        //System.out.println(com.compare("圆1", "圆2"));
        //编译错误,因为"圆1", "圆2"不是Circle类型,是String类型,编译器提前报错,
        //而不是冒着风险在运行时再报错。
    }
}

CompareTor构造器的定义方式:使用泛型实现,指定只接受某个类型的
只有比较的定义不一样,其它的部分都是一样的,这里以匿名内部类方式

//价格从小打到排序,价格一样,name从大到小排序
Comparator<Product> comp = new Comparator<Product>() {
    @Override
    public int compare(Product p1, Product p2) {
        int res = Double.compare(p1.price,p1.price);
        if(res != 0){
            return res;//从小到大排序
        }
        return -p1.name.compareTo(p2.name);//从大到小排序
    }
};

1.2 自定义范型结构(类/接口/方法)

1、<类型>这种语法形式就叫泛型。
<T>:称为类型参数,代表未知的数据类型,可以指定为<String>,<Integer>,<Circle>等。类比方法中形参将<T>称为类型形参,将<Circle>称为类型实参。T可以替换成K,V等任意字母

泛型类/接口的声明:在类名或接口名后面声明泛型类型,我们把这样的类或接口称为泛型类泛型接口
多个泛型参数声明:<E1,E2,E3>
JDK7.0 开始,泛型的简化操作:ArrayList<Fruit> flist = new ArrayList<>();

【修饰符】 class 类名<类型变量列表>extends 父类】 【implements 接口们】{
    
}
【修饰符】 interface 接口名<类型变量列表>implements 接口们】{
    
}
//例如:
public class ArrayList<E>    
public interface Map<K,V>{
    ....
}    

泛型方法声明格式:在【修饰符】与返回值类型之间声明类型变量。把声明了类型变量的方法,称为泛型方法。

[修饰符] <类型变量列表> 返回值类型 方法名([形参列表])[throws 异常列表]{
    //...
}
public class DAO {
    public <E> E get(int id, E e) {
        E result = null;
        return result;
    }
}

1、自定义泛型类或泛型接口、方法:

  • 泛型类或接口:在类或接口中定义某个成员时,该成员的相关类型是不确定的,而这个类型需要在使用这个类或接口时才可以确定
    • 在类的内部(比如:属性、方法、构造器中)使用类的泛型。其在本类或本接口中即代表某种类型
    • 创建自定义泛型类的对象时:
      • 可指明泛型参数类型。一旦指明,内部凡是使用类的泛型参数的位置,都具体化为指定的类的泛型类型。
      • 泛型擦除:如果没有指明泛型参数类型,那么泛型对应的类型均按照Object处理,但不等价于Object。
    • 指定泛型类型:只能是引用数据类型,如果是基本则可以使用包装类型代替
    • 如果泛型结构是一个接口或抽象类,则不可创建泛型类的对象。
    • 不可以在静态方法中使用类的泛型。
    • 异常类不能是带泛型的
    • 不能用new E[],但可E[] elements = (E[])new Object[capacity];参考ArrayList源码声明:Object[] elementData,而非泛型参数类型数组。
  • 泛型方法:在没有定义泛型类/接口时,但是某个方法形参类型不确定时,可以单独定义<泛型参数>。
    • 方法可以被泛型化,并且可以被声明为static的,与所在类是否泛型无关,其中的泛型参数在方法被调用时确定。

2、泛型类/接口继承的一些特性

  • 泛型类型的继承性:继承/实现泛型类/接口时,子类/接口可以确定泛型结构中的泛型参数。如果不确定泛型的类型,则可以继续使用泛型参数,还可以新增泛型参数。
  • 如果B是A的一个子类型(子类或者子接口),而G是具有泛型声明的类或接口,G<B>并不是G<A>的子类型!
    比如:String是Object的子类,但是List<String>并不是List<Object>的子类。

泛型继承的一些特性

class A<T>{
}
//1)没有类型 擦除
class B extends A{}
//子类:继承父类的泛型参数// 1)全部保留
class B extends A<T>{
}
//子类:指定父类的泛型参数类型 // 2)具体类型
class C extends A<Integer>{
}
//子类:在父类的基础上新增泛型参数
class D<E> extends A<Integer>{
}

泛型类/接口、方法的一些实现(及泛型方法的调用方式)

@Test
public void test8(){
	//T[] arr = new T[];//是错误的写法
	//T[] arr = (T[])new Object[6];
    Integer[] arr = new Integer[]{12,21,43,56};
    Integer[] arr1 = reverse(arr);
    for (int i = 0; i < arr1.length; i++) {
        System.out.println(arr1[i]);
    }

} //编写一个泛型方法,接收一个任意引用类型的数组,并反转数组中的所有元素
public <T> T[] reverse(T[] arr){
    for (int min = 0,max = arr.length-1; min <max; min++,max--) {
        T temp = arr[min];
        arr[min] = arr[max];
        arr[max] = temp;
    }
    return arr;
}

泛型类的使用场景:声明一个学生类,该学生包含姓名、成绩,而此时学生的成绩类型不确定,为什么呢,因为,语文老师希望成绩是“优秀”、“良好”、“及格”、“不及格”,数学老师希望成绩是89.5, 65.0,英语老师希望成绩是’A’,‘B’,‘C’,‘D’,‘E’。那么我们在设计这个学生类时,就可以使用泛型。

//省略了get/set方式
class Student<T>{
    String name;
    T score;

    public Student() {
        super();
    }
    public Student(String name, T score) {
        super();
        this.name = name;
        this.score = score;
    }
    @Override
    public String toString() {
        return "姓名:" + name + ", 成绩:" + score;
    }
}
public class TestStudent {
    public static void main(String[] args) {
        //语文老师使用时:
        Student<String> stu1 = new Student<String>("张三", "良好");

        //数学老师使用时:
        //Student<double> stu2 = new Student<double>("张三", 90.5);//错误,必须是引用数据类型
        Student<Double> stu2 = new Student<Double>("张三", 90.5);

        //英语老师使用时:
        Student<Character> stu3 = new Student<Character>("张三", 'C');

        //错误的指定
        //Student<Object> stu = new Student<String>();//错误的
    }
}

二、通配符&读写特点

1、通配符:?
注意:不能用在泛型方法声明上;不能用在泛型类的声明上;不能用在创建对象
2、通配符的使用:以ArrayList<?>为例 G\<?> 可以看做是G<A>类型的父类,G<?>引用/变量=G<A>对象

读写数据的特点(以集合ArrayList<?>为例说明)
读取数据:允许的,读取的值的类型为Object类型
写入数据:不允许的。特例写入null值。

4、有限制条件的通配符
List<? extends A>:理解"<=“(-∞,A],可以将List<A>或List<B>赋值给List<? extends A>。其中B类是A类的子类。
List<? super A>:理解”>="[A,+∞),可以将List<A>或List<B>赋值给List<?extends A>。其中B类是A类的父类。

读写数据的特点:技巧,开发中编译是否报错
? extends A:以集合为例:可以读取数据、不能写入数据(例外:null)
? super A:以集合为例:可以读取数据、可以写入A类型或A类型子类的数据(例外:null)

通配符的使用

//注意点1:编译错误:不能用在泛型方法声明上,返回值类型前面<>不能使用?
public static <?> void test(ArrayList<?> list){
}
//注意点2:编译错误:不能用在泛型类的声明上
class GenericTypeClass<?>{
}
//注意点3:编译错误:不能用在创建对象上,右边属于创建集合对象
ArrayList<?> list2 = new ArrayList<?>();

有限制的通配符

<? extends Number>     //(无穷小 , Number]
//只允许泛型为Number及Number子类的引用调用

<? super Number>      //[Number , 无穷大)
//只允许泛型为Number及Number父类的引用调用

<? extends Comparable>
//只允许泛型为实现Comparable接口的实现类的引用调用

测试通配符及有限制条件的通配符、读写特点

//测试:通配符?的使用
@Test
public void test1() {
    List<?> list = null;
    List<Object> list1 = null;
    List<String> list2 = null;
    list = list1;
    list = list2;
    method(list1);
    method(list2);
}
public void method(List<?> list) {
    for (Object obj : list) {
        System.out.println(obj);
    }
    //list.add("AA");//错误的
}
//测试:通配符?的读写数据的特点
@Test
public void test2() {
    List<?> list = null;
    List<String> list1 = new ArrayList<>();
    list1.add("AA");
    list = list1;
    //读取数据(以集合为例说明)
    Object obj = list.get(0);
    System.out.println(obj);
    //list.add("BB");//写入数据(以集合为例说明)//写入数据,操作失败。
    //特例:可以将null写入集合中。
    list.add(null);
}
//有限制条件的通配符的使用:? extends Father。范围(无穷小,Father]
@Test
public void test3(){
    List<? extends Father> list = null;
    List<Object> list1 = null;
    List<Father> list2 = null;
    List<Son> list3 = null;
    //list = list1;//编译错误,?不包括这个范围
    list = list2;
    list = list3;
}
//针对于? extends A的读写
@Test
public void test3_1() {
    List<? extends Father> list = null;
    List<Father> list1 = new ArrayList<>();
    list1.add(new Father());
    list = list1;
    //读取数据:可以的Father father = list.get(0);
    //写入数据:
    list.add(null);
    //list.add(new Father());
    //list.add(new Son());//编译错误,还可能是更小的
}
//有限制条件的通配符的使用:? super Father。范围[Father,无穷大)
@Test
public void test4() {
    List<? super Father> list = null;
    List <Object > list1 = null;
    List<Father> list2 = null;
    List<Son> list3 = null;
    list = list1;
    list = list2;
    //list = list3;//不包括这个范围
}
public void test4_1() {
    List<? super Father> list = null;
    List<Father> list1 = new ArrayList<>() list1.add(new Father());
    list = list1;
    //读取数据:可以的
    Object obj = list.get(0);
    //写入数据:
    list.add(null);
    //list.add(new Object());
    list.add(new Father());//多态的特性,小的可以添加给大的
    list.add(new Son());
}

三、企业真题

1、Java 的泛型是什么?有什么好处和优点?JDK 不同版本的泛型有什么区别?(软动力)
泛型,是程序中出现的不确定的类型。
以集合来举例:把一个集合中的内容限制为一个特定的数据类型,这就是generic背后的核心思想。
jdk7.0新特性:

ArrayList<String> list = new ArrayList<>(); //类型推断

后续版本的新特性:

Comparator<Employee> comparator = new Comparator<>(){} //类型推断

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/595044.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Android getevent命令详细分析

在调试Android 的输入事件时&#xff0c;经常使用 “getevent -lrt” 命令&#xff0c;来确认驱动上报数据是否正常。从源码的角度来详细的分析一下getevent 这个程序。 首先用ls命令来看一下getevent lrwxr-xr-x 1 root shell 7 2023-11-20 10:08 system/bin/getevent -> …

学习java中的interface接口

1.了解接口 java提供了一个关键字interface&#xff0c;用这个关键字我们可以定义出一个特殊的结构&#xff1a;接口 格式&#xff1a; public interface 接口名{ //成员变量&#xff08;常量&#xff09; //成员方法&#xff08;抽象方法&#xff09; } 注意&#xff1a;接…

cmake进阶:宏定义

一. 简介 前面学习了 CMakeLists.txt语法中是如何定义函数&#xff0c;本文继续学习 cmake中的宏定义。 二. cmake进阶&#xff1a;宏定义 cmake 提供了定义宏的方法&#xff0c;cmake 中函数 function 和宏定义 macro 在某种程度上来说是一样的&#xff0c;都是创建一段有…

Linux 内核简介

操作系统简介 操作系统概念&#xff1a;操作系统处于硬件和应用程序的中间层&#xff0c;控制和管理整个计算机系统的硬件和软件资源&#xff0c;提供给用户和其他软件方便的接口和环境&#xff0c;它是计算机系统的最基本的系统软件。 操作系统功能: 处理机管理存储器管理设…

硬件四舍五入模式

文章目录 舍入模式1. 就近舍入2.向0舍入3.远离0舍入4. 向正无穷舍入5. 向负无穷舍入6.向负无穷舍入7. ROUND TO MATH参考资料 舍入模式 为了减小舍入操作对推理结果的精度影响&#xff0c;AI芯片有时会支持多种舍入模型(round mode)供编程人员选择&#xff0c;常见模式如下表&…

websevere服务器从零搭建到上线(二)|Linux上的五种IO模型

文章目录 阻塞 blocking非阻塞 non-blockingIO复用 IO multiplexing信号驱动 signal-driven异步 asynchronous拓展知识 看过上篇文章英国基本能理解本文五张图的内容websevere服务器从零搭建到上线&#xff08;一&#xff09;&#xff5c;阻塞、非阻塞、同步、异步 本文要能够在…

IP地址类型

这些IP地址中有IPv4和IPv6地址,以及一些是链路本地地址(通常用于本地网络中的通信),而另一些则是可以被路由的公共或私有IP地址。 这里是您提供的IP地址的一些简要说明: IPv6 链路本地地址:以fe80:开头的地址是IPv6链路本地地址。这些地址仅用于同一链路(如以太网段或无…

Verilog中求两个数的差值

根据输入信号a,b的大小关系&#xff0c;求解两个数的差值&#xff1a;输入信号a,b为8bit位宽的无符号数。如果a>b&#xff0c;则输出a-b&#xff0c;如果a≤b&#xff0c;则输出b-a。 接口信号图如下&#xff1a; 代码如下&#xff1a; &#xff08;CSDN代码块不支持Veril…

SpringBoot 使用 @RequiredArgsConstructor(onConstructor_ = @Autowired) 报错解决

若使用 RequiredArgsConstructor(onConstructor_ Autowired) 启动报错&#xff0c;或者爆红可以使用以下方法解决 1. 安装或启用 Lombok插件 2. 检查 Lombok 版本 3. 若 onConstructor_ 爆红&#xff0c; 可能是IDEA中文软件包冲突 4. 若以上还是不行&#xff0c;可以添加…

数据结构-二叉树-二叉搜索树

一、概念 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者具有以下性质的二叉树&#xff1a; 若它的左子树不为空&#xff0c;则左树上所有节点的值都小于根节点的值。 若它的右子树不为空&#xff0c;则右子树上所有节点的值都大于根节点的值。 它…

27.leetcode---随机链表的复制(Java版)

题目链接: https://leetcode.cn/problems/copy-list-with-random-pointer/description/ 题目解析: 使用map来解这个题就比较方便了 代码: 测试:

手机异地组网方案?

现代社会&#xff0c;随着信息技术的快速发展&#xff0c;人们之间的通信需求也日益增加。尤其是在异地工作、异地学习、异地旅游等情况下&#xff0c;我们需要实现不同地区间的快速组建局域网&#xff0c;以解决电脑与电脑、设备与设备、电脑与设备之间的信息远程通信问题。本…

深度学习中的注意力机制一(Pytorch 15)

一 简介 灵长类动物的视觉系统接受了大量的感官输入&#xff0c;这些感官输入远远超过了大脑能够完全处理的程度。然而&#xff0c; 并非所有刺激的影响都是相等的。意识的聚集和专注使灵长类动物能够在复杂的视觉环境中将注意力引向感 兴趣的物体&#xff0c;例如猎物和天敌。…

C语言 函数递归

函数递归 一、递归是什么&#xff1f; 递归是学习C语⾔函数绕不开的⼀个话题&#xff0c;那什么是递归呢&#xff1f; 递归其实是⼀种解决问题的⽅法&#xff0c;在C语⾔中&#xff0c;递归就是函数⾃⼰调⽤⾃⼰。 写⼀个史上最简单的C语⾔递归代码&#xff1a; #include <…

量水堰计使用手册:流量监测的关键工具

在水资源管理、环境监测、工业应用中&#xff0c;精确测量流体流量是至关重要的。量水堰计是一种流行且有效的工具&#xff0c;用于监测开放水道中的水流量。本文将详细介绍量水堰计的原理、安装、使用方法及常见问题解决策略&#xff0c;旨在帮助用户更好地理解和运用该设备。…

Python 机器学习 基础 之 构建第一个机器学习应用

Python 机器学习 基础 之 构建第一个机器学习应用 目录 Python 机器学习 基础 之 构建第一个机器学习应用 一、简单介绍 二、第一个机器学习测试应用介绍&#xff1a;鸢尾花分类 三、第一个机器学习测试应用 &#xff1a;前置环境&#xff0c;知识点介绍 jupyter notebo…

案例精选 | 江苏省徐州医药高等职业学校网络流量纵深防护项目

江苏省徐州医药高等职业学校&#xff0c;自1985年建校以来&#xff0c;始终站在医药教育的前沿。作为江苏省药品监督管理局直属的高等职业学校&#xff0c;该校不仅是江苏省医药行业协会的重要成员&#xff0c;还是中国职业技术教育学会医药专业委员会的副主任单位。 学校拥有…

C++:重写和重载

重写&#xff08;Override&#xff09;和重载&#xff08;Overload&#xff09;是面向对象编程中常用的两个概念&#xff0c;它们虽然都涉及到方法的定义&#xff0c;但是在实现和使用上有着不同的特点。 重写&#xff08;Override&#xff09;&#xff1a; 重写是指在子类中重…

Web安全研究(九)

知识星球 首先推荐一下我们的知识星球,以AI与安全结合作为主题,包括AI在安全上的应用和AI本身的安全; 加入星球你将获得: 【Ai4sec】:以数据驱动增强安全水位,涵盖内容包括:恶意软件分析,软件安全,AI安全,数据安全,系统安全,流量分析,防爬,验证码等安全方向。…

linux安装 mysql

环境&#xff1a;centOS8 一、安装 1 安装wget库 sudo yum -y install wget 2. 安装 mysql 换yum源 亲测成功&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 换yum源 1.下载对应版本的repo文件 wget -O CentOS-Base.repo http://mirrors…
最新文章