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是调用方法的时候确定的,并非实例化类的时候确定,所以和类的泛型没关系,所以泛型方法可以是静态的
}
目录
相关文章
|
1月前
|
消息中间件 Java Kafka
在Java中实现分布式事务的常用框架和方法
总之,选择合适的分布式事务框架和方法需要综合考虑业务需求、性能、复杂度等因素。不同的框架和方法都有其特点和适用场景,需要根据具体情况进行评估和选择。同时,随着技术的不断发展,分布式事务的解决方案也在不断更新和完善,以更好地满足业务的需求。你还可以进一步深入研究和了解这些框架和方法,以便在实际应用中更好地实现分布式事务管理。
|
1月前
|
Java
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
74 9
|
16天前
|
Java
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
79 34
|
2天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
1月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
46 8
|
28天前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
41 4
|
28天前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
28 4
|
26天前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
26 1
|
1月前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
139 57