Java泛型02:自定义泛型类、泛型方法

简介: Java泛型02:自定义泛型类、泛型方法

一、自定义泛型类(接口)

@[toc]
ps:泛型类和泛型接口的区别就是类和接口的区别,这里不做阐述

1、基础知识

  1. 泛型类可能有多个参数,此时应将多个参数一起放在尖括号内。比如:<E1,E2,E3>
  2. 泛型类的构造器如下:public GenericClass(){}。 而下面是错误的:public GenericClass(){}
  3. 实例化后,操作原来泛型位置的结构必须与指定的泛型类型一致
  4. 泛型不同的引用不能相互赋值。(见下面代码举例部分的第四点测试类的)
  5. 泛型如果不指定,将被擦除,泛型对应的类型均按照Object处理,但不等价 于Object。经验:泛型要使用一路都用。要不用,一路都不要用。
  6. 如果泛型结构是一个接口或抽象类,则不可创建泛型类的对象。
  7. jdk1.7,泛型的简化操作:ArrayList flist = new ArrayList<>();
  8. 泛型的指定中不能使用基本数据类型,可以使用包装类替换。
  9. 在类/接口上声明的泛型,在本类或本接口中即代表某种类型,可以作为非静态属性的类型、非静态方法的参数类型、非静态方法的返回值类型。但在静态方法中不能使用类的泛型。(看下面代码举例第一点自定义泛型类Order注释部分)(原因:泛型在创建对象的时候才指定,但是静态方法要早于创建对象,所以静态方法中不能使用类的泛型)
  10. 异常类不能是泛型的

    public class Test11<T> extends Exception{}//错误的
  11. 不能使用new E[]。但是可以:E[] elements = (E[])new Object[capacity];

    参考:ArrayList源码中声明:Object[] elementData,而非泛型参数类型数组。

    举例:

    T[] t = new T[];//错误的
    T[] t = (T[]) new Object[10];//而且后续用的时候赋值的时候new的只能是T或者T的子类的对象,不然会报错
  12. .父类有泛型,子类可以选择保留泛型也可以选择指定泛型类型:

    • 子类不保留父类的泛型:按需实现

      • 没有类型 擦除
      • 具体类型
    • 子类保留父类的泛型:泛型子类

      • 全部保留
      • 部分保留

代码举例:

class Father<T1, T2> {
        }
// 子类不保留父类的泛型
// 1)没有类型 擦除
        class Son1 extends Father {// 等价于class Son extends Father<Object,Object>{
        }
// 2)具体类型
        class Son2 extends Father<Integer, String> {
        }
// 子类保留父类的泛型
// 1)全部保留
        class Son3<T1, T2> extends Father<T1, T2> {
        }
// 2)部分保留
        class Son4<T2> extends Father<Integer, T2> {
        }

class Father<T1, T2> {
        }
// 子类不保留父类的泛型
// 1)没有类型 擦除
        class Son<A, B> extends Father{//等价于class Son extends Father<Object,Object>{
        }
// 2)具体类型
        class Son2<A, B> extends Father<Integer, String> {
        }
// 子类保留父类的泛型
// 1)全部保留
        class Son3<T1, T2, A, B> extends Father<T1, T2> {
        }
// 2)部分保留
        class Son4<T2, A, B> extends Father<Integer, T2> {
        }

2、代码举例

  1. 自定义泛型类Order

    package com.jsm.java1;
    //自定义泛型类
    public class Order<T> {
        String orderName;
        int orderId;
    
        //类的内部结构就可以使用类的泛型
        T orderT;
        public Order(){
    
        }
        public Order(String orderName, int orderId, T orderT) {
            this.orderName = orderName;
            this.orderId = orderId;
            this.orderT = orderT;
        }
    
        public T getOrderT() {
            return orderT;
        }
    
        public void setOrderT(T orderT) {
            this.orderT = orderT;
        }
    
        @Override
        public String toString() {
            return "Order{" +
                    "orderName='" + orderName + '\'' +
                    ", orderId=" + orderId +
                    ", orderT=" + orderT +
                    '}';
        }
        //静态方法中不能使用类的泛型。
        /*
        public static void show(T orderT){
            System.out.println(T orderT);
        }
         */
        
        
    }
  2. 继承泛型类Order并且指明了泛型为Integer(SubOrder不再是泛型类

    public class SubOrder extends Order<Integer> {
    }
  3. 继承泛型类Order但是仍然保留了”T”(SubOrder1仍然是泛型类

    public class SubOrder1<T> extends Order<T> {
    }
  4. 测试:OrderTest

    package com.jsm.java1;
    
    import org.junit.Test;
    
    public class OrderTest {
        @Test
        public void test1(){
            //如果定义了泛型类,实例化没有指明类的泛型,则认为此泛型类型为Object
            //要求:如果定义了类是带泛型的,建议在实例化时候要指明类的泛型
            Order<Object> order = new Order<>();
            order.setOrderT(123);
            order.setOrderT("ABC");
    
    
            //建议:实例化时候指明类的泛型
            Order<String> order1 = new Order<String>("orderAA",1001,"F");
            order1.setOrderT("AA:hello");
    
            SubOrder sub1 = new SubOrder();
            //由于子类在继承带父类的泛型时,指明了泛型类型,则实例化子类对象时,不再需要指明泛型
            sub1.setOrderT(1234);
            
            SubOrder1<Integer> sb1 = new SubOrder1<Integer>();
            sb1.setOrderT(1234);
            
            
             @Test
        public void test2(){
            //泛型不同的引用不能相互赋值
            ArrayList<String> list1= new ArrayList<String>();
            ArrayList<Integer> list2= new ArrayList<Integer>();
            //list1=list2;
            //尽管在编译时ArrayList<String>和ArrayList<Integer>是两种类型,但是,在运行时只有
            //一个ArrayList被加载到JVM中。
    
    
        }
    
    
        }
    }

二、自定义泛型方法

1、基础知识

  1. 在方法中出现了泛型结构、泛型参数与类的泛型参数没有任何关系,换句话说,是不是泛型方法,和泛型方法所属的类是不是泛型类没有任何关系
  2. 举例代码中方法这里的E是调用方法的时候确定的,并非实例化类的时候确定,所以和类的泛型没关系,所以泛型方法可以是静态的
  3. 泛型方法的格式:

    [访问权限] <泛型> 返回类型 方法名([泛型标识 参数名称]) 抛出的异常

2、代码举例

//学习内容:
//开发时间:10月16日  0:13
package com.jsm.java2;

import org.junit.Test;

import java.util.ArrayList;
import java.util.List;

public class MethodTest {
    //测试泛型方法
    @Test
    public void test11(){
        Integer[] arr=new Integer[]{1,2,3,4};
        //泛型方法在调用时,指明泛型参数的类型,这个类型和类的泛型没有任何关系
        List<Integer> list = copyFromArrayToList(arr);
        System.out.println(list);//[1, 2, 3, 4]
    }
    //泛型方法
    public  <E> List<E> copyFromArrayToList(E[] arr){
        ArrayList<E> list=new ArrayList<>();
        for (E e:arr){
            list.add(e);
        }
        return list;
    }
    //说明:这个方法这里的E是调用方法的时候确定的,并非实例化类的时候确定,所以和类的泛型没关系,所以泛型方法可以是静态的
}
目录
相关文章
|
4天前
|
存储 Java 索引
Java快速入门之数组、方法
### Java快速入门之数组与方法简介 #### 一、数组 数组是一种容器,用于存储同种数据类型的多个值。定义数组时需指定数据类型,如`int[]`只能存储整数。数组的初始化分为静态和动态两种: - **静态初始化**:直接指定元素,系统自动计算长度,如`int[] arr = {1, 2, 3};` - **动态初始化**:手动指定长度,系统给定默认值,如`int[] arr = new int[3];` 数组访问通过索引完成,索引从0开始,最大索引为`数组.length - 1`。遍历数组常用`for`循环。常见操作包括求和、找最值、统计特定条件元素等。
|
1天前
|
安全 Java 程序员
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
29 12
|
2天前
|
算法 Java API
Java 方法注释:规范、实用和高质量的写法
本文深入探讨了如何编写高质量的 Java 方法注释
25 11
|
2天前
|
SQL Java 数据库连接
【潜意识Java】Java中JDBC过时方法的替代方案以及JDBC为什么过时详细分析
本文介绍了JDBC中一些常见过时方法及其替代方案。
21 5
|
1月前
|
Java
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
103 34
|
29天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
2月前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
60 4
|
2月前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
32 1
|
7月前
|
Java API 容器
Java泛型的继承和通配符
Java泛型的继承和通配符
45 1
|
8月前
|
安全 Java API
Java一分钟之-泛型通配符:上限与下限野蛮类型
【5月更文挑战第19天】Java中的泛型通配符用于增强方法参数和变量的灵活性。通配符上限`? extends T`允许读取`T`或其子类型的列表,而通配符下限`? super T`允许向`T`或其父类型的列表写入。野蛮类型不指定泛型,可能引发运行时异常。注意,不能创建泛型通配符实例,也无法同时指定上下限。理解和适度使用这些概念能提升代码的通用性和安全性,但也需兼顾可读性。
77 3