Java泛型

Java泛型是一种类型安全机制,允许在编译时检查类型,防止不匹配的数据类型存入集合。泛型可以应用于类、接口和方法,提供了一种方式来指定容器可以持有哪种类型的元素,避免了强制类型转换。类型擦除意味着在运行时,泛型信息会丢失,所有泛型类最终都会退化为Object类型或其上限类型。
摘要由CSDN通过智能技术生成

什么是泛型

泛型,即“参数化类型” 。一提到参数,最熟悉的就是定义方法时有形参,然后调用此方法时传递实参。

参数化类型,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式,然后在使用/调用时传入具体的类型。

Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,泛型的好处就是在编译的时候能够检查类型安全。

import java.util.ArrayList;
import java.util.Date;
public class Demo1 {
	 public static void main(String[] args) {
		/*
		   ArrayList<E>  E-->参数-->类型
		                泛型--->参数化类型,类型参数化
		                
		   java中所有的集合类,都支持泛型,支持向类中,传入一个类型,
		         建议集合中只存储一种类型 , 就不需要进行向下转型          
		 */
	/* 
	    不使用泛型的反例
            ArrayList list = new ArrayList<>();
            list.add("aaa");
            list.add("aaa");
            list.add(new Date());
            list.add(10);
            for(Object obj : list){
                if(obj instanceof String){
                    String s = (String)obj;
                    s.length();
                }
                if(obj instanceof Date){
                    String s = (String)obj;
                    s.length();
                }
            }  */
		 //使用泛型的案例
	 ArrayList<String> list = new ArrayList<>();
         list.add("aaa");
         list.add("aaa");
         for(String obj : list){
              String s = (String)obj;
              s.length();
        }  
	}
}

泛型类

泛型类型用于类的定义中,被称为泛型类。通过泛型可以完成对一组类的操作对外开放相同的接口。

一个最普通的泛型类:

public class Car<T> {//T可以为任意标识符,常见的如T(type)、E(element)、K(key)、V(value)等形式的参数常用于表示泛型,类型是和创建对象时的类型所对应的
    private T name;//成员变量的类型为T,T的类型由外部指定
    //private int price;
    
    public Car() {
    }

    public Car(T name) {//泛型构造方法形参name的类型也为T,T的类型由外部指定
        this.name = name;
    }

    public T getName() {//泛型方法getName的返回值类型为T,T的类型由外部指定
        return name;
    }

    public void setName(T name) {
        this.name = name;
    }

    public static void main(String[] args) {
        //类型参数化, 把类型当做参数一样,动态的传递
        Car<String> car1=new Car<>("aaa");//传入的实参类型需与泛型的类型参数类型相同,即为String
    }
}

1.泛型的类型参数只能是类类型 ,不能是int类型。。。。

2.泛型的类型参数可以有多个。

3.如果没有定义具体类型,默认为Object

从泛型类派生子类

1.子类也是泛型类,子类和父类的泛型类型要一致

class A<T> extends Demo<T>

2.子类不是泛型类,父类要明确泛型的数据类型

class A extends Demo<String>

泛型接口

泛型接口与泛型类的定义及使用基本相同

public interface Demo<T> { //定义一个泛型接口 }

1.子类也是泛型类,子类和父类的泛型类型要一致

class A<T> implements Demo<T> { }

//如果一个子类继承/实现父级类/接口是有泛型的,那么子类也可以定义为一个泛型类
public class Person<T> implements Comparable<T>{
	private String name;
	@Override
	public int compareTo(Person o) {
		return 0;
	}
	public static void main(String[] args) {
		Person<String> p =   new Person<>();
		p.compareTo("aaa");
	}
}

2.子类不是泛型类,父类要明确泛型的数据类型

public class A implements Demo<String> { }

//如果一个子类继承/实现父级类/接口是有泛型的,那么子类也可以不定义为泛型类,那么父类/接口类型必须要明确才行
public class Person implements Comparable<Person>{
	 private String name;
	@Override
	public int compareTo(Person o) {
		return 0;
	}
	public static void main(String[] args) {
		Person p = new Person();
	}
}

泛型通配符

类型通配符 一般是使用"?",代替具体的类型实参,表示实际传入的参数类型可以是任意的

public class Demo<T>{ 
	//test(Demo<?> d)  ? 类型通配符,指的是实际传入参数的类型, 可以是任何类型
	 public void test(Demo<?> d){
         
	 }
	 public static void main(String[] args) {
		  Demo d0 = new Demo<>();
		  Demo<Integer> d1 = new Demo<>();
		  Demo<String> d2 = new Demo<>();
		  d0.test(d1);
		  d0.test(d2);
	}
}

<? extends T> 泛型上限,实际传入参数的泛型,上限是T 以及T的子类

<? super T> 泛型下限,实际传入参数的泛型,下限是T 以及T的父类

public class Demo<T>{
	 public void test(Demo<?> d){
		 
	 }
      public void test1(Demo<? extends T> d){//传进来的类型比T小,或者等于T
		 
	 }
      public void test2(Demo<? super T> d){//传进来的类型比T大,或者等于T
 		 
 	 }
	 
	 public static void main(String[] args) {
		 
		  /*Demo<Number> d0 = new Demo<>();
		  Demo<Integer> d1 = new Demo<>();//Integer继承了Number
		  Demo<Number> d2 = new Demo<>();
		  Demo<String> d3 = new Demo<>();
		    d0.test1(d1);
		    d0.test1(d2);
		    d0.test1(d3);//错误❌
		    */
		 
		  Demo<Number> d0 = new Demo<>();
		  Demo<Object> d1 = new Demo<>();
		  Demo<Number> d2 = new Demo<>();
		  d0.test2(d1);
		  d0.test2(d2);
	}
}

类型擦除

泛型是Java 1.5版本才引进的概念,在这之前是没有泛型的,但是,泛型代码能够很好地和之前版本的代码兼容。那是因为,泛到信息只存在于代码编译阶段,在进入JVM之前,与泛型相关的信息会被擦除掉,我们称之为一类型擦除。泛型类被类型擦除后,相应的类型就被替换成 Object 类型或者上限类型。

泛型只是在编译期间来对类型进行限制,底层还是Object类型,这就是泛型类型擦除。

import java.lang.reflect.Field;
import java.util.ArrayList;
public class Demo2<T> {
    public T name;
    public static void main(String[] args) throws NoSuchFieldException, SecurityException {
        // transient Object[] elementData; 
        ArrayList<String> list  =  new ArrayList();
        list.add("");

        Demo2<String> d2 = new Demo2<>();
        d2.name = "aaa";
         //同过反射机制,获取运行时的属性,结果发现,运行时,底层是Object类型             
         Class c =   d2.getClass();//把Demo2对象在内存中的类地址,类信息拿到了   
         Field f =  c.getField("name");
         System.out.println(f.getName()+":::"+f.getType());//name:::class java.lang.Object

    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值