Java泛型理解

前言:

  当元素存入集合时,集合会将元素转换为Object类型存储,当取出时也是按照Object取出的,所以用get方法取出时,我们会进行强制类型转换,并且通过代码也可以看出来,我们放入其他类型时,如字符串,编译器不会报错,但是运行程序时会抛出类型错误异常,这样给开发带来很多不方便,用泛型就解决了这个麻烦。

  泛型规定了某个集合只能存放特定类型的属性,当添加类型与规定不一致时,编译器会直接报错,可以清楚的看到错误。

  当我们从List中取出元素时直接取出即可,不用类型转换,因为已经规定了里面存放的只能是某种类型。

  集合中中除了存入定义的泛型类型的实例,还可以存入泛型类型子类的实例。

  泛型不能是基本类型,只能是引用类型,如果必须使用基本类型,可以使用基本类型的包装类。

一.泛型是什么?

 Java 泛型是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。
泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。
通俗的讲,泛型就是操作类型的 占位符,即:假设占位符为T,那么此次声明的数据结构操作的数据类型为T类型。

二.泛型方法

  • 所有泛型方法声明都有一个类型参数声明部分(由尖括号分隔),该类型参数声明部分在方法返回类型之前(在下面例子中的)。
  • 每一个类型参数声明部分包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符。
  • 类型参数能被用来声明返回值类型,并且能作为泛型方法得到的实际参数类型的占位符。
  • 泛型方法体的声明和其他方法一样。注意类型参数只能代表引用型类型,不能是原始类型(像int,double,char的等)。

下面给出使用不同类型数组打印元素的实例

public class GenericMethodTest {
	public static <E> void printArray(E[] arrays) {
		for(E e : arrays) {
			System.out.print(e+" ");
		}
		System.out.println();
	}
	public static void main(String[] args) {
		Integer[] intArray = {10,20,33,14,53};
		Double[] doubleArray = {53.1,25.6,56.8,22.1,25.3};
		String[] stringArray = {"Apple","Banana","Watermelon","Strawberry"};
		
		System.out.println("整型数组元素为:");
		printArray(intArray);
		
		System.out.println("双精度浮点数数组元素为:");
		printArray(doubleArray);
		
		System.out.println("字符串数组元素为:");
		printArray(stringArray);
	}
}

运行结果如下:

整型数组元素为:
10 20 33 14 53 
双精度浮点数数组元素为:
53.1 25.6 56.8 22.1 25.3 
字符串数组元素为:
Apple Banana Watermelon Strawberry 
有界参数类型:

一个操作数字的方法可能只希望接收Number以及Number的子类实例。为了限制传递类型参数的范围,我们可以声明一个有界的类型参数,如何列出类型参数的名称,后接extends和他的上界。
下面我们使用一个例子来说明:

public static <T extends Comparable<T>> T maxnum(T x,T y,T z) {
		T max= x;
		if(y.compareTo(max)>0) {
			max = y;
		}
		if(z.compareTo(max)>0) {
			max = z;
		}
		return max;
	}
	public static void main(String[] args) {
		System.out.println("85, 55, 54中最大的数字为:"+maxnum(85, 55, 54));
		
		System.out.println("apple watermelon orange中最大的字符为:" + maxnum("apple", "watermelon", "orange"));
		
	}

二.泛型类

  泛型类的类型参数声明部分也包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符。因为他们接受一个或多个参数,这些类被称为参数化的类或参数化的类型。
接下来我们演示一个实例:

public class Box<T>{
	private T t;
	public void add(T t) {
		this.t = t;
	}
	public T get() {
		return t;
	}
	
	public static void main(String[] args) {
		Box<Integer> integBox = new Box<Integer>();
		Box<String> stringBox = new Box<String>();
		
		integBox.add(88);
		stringBox.add("Hello World!");
		
		System.out.println("整形值为:" + integBox.get());
		System.out.println("字符串为:" + stringBox.get());
	}
}

运行结果:

整形值为:88
字符串为:Hello World!

三.类型通配符

类型通配符一般是使用?代替具体的类型参数。例如 List<?> 在逻辑上是List,List 等所有List<具体类型实参>的父类。

public class GenericTest {
	public static String getData(List<?> data) {
		return "data:" + data.get(0);
	}
	public static void main(String[] args) {
		List<String> course = new ArrayList<String>();
		List<Integer> no = new ArrayList<Integer>();
		List<Number> num = new ArrayList<Number>();
		
		course.add("CS");
		no.add(19);
		num.add(30);
		
		System.out.println(getData(course));

		System.out.println(getData(no));

		System.out.println(getData(num));
	}
	
}

运行结果:

data:CS
data:19
data:30

四.Java泛型实现原理:类型擦除

  Java的泛型是伪泛型。在编译期间,所有的泛型信息都会被擦除掉。正确理解泛型概念的首要前提是理解类型擦出(type erasure)。

  Java中的泛型基本上都是在编译器这个层次来实现的。在生成的Java字节码中是不包含泛型中的类型信息的。使用泛型的时候加上的类型参数,会在编译器在编译的时候去掉。这个过程就称为类型擦除。

   如在代码中定义的List和List等类型,在编译后都会变成List。JVM看到的只是List,而由泛型附加的类型信息对JVM来说是不可见的。Java编译器会在编译时尽可能的发现可能出错的地方,但是仍然无法避免在运行时刻出现类型转换异常的情况。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值