java高级特性之泛型与通配符

参考传智播客视频

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

1. 泛型类:具有一个或多个类型变量的类,称之为泛型类!



class A<T> {
}


2. 在创建泛型类实例时,需要为其类型变量赋值


A<String> a = new A<String>();
  * 如果创建实例时,不给类型变量赋值,那么会有一个警告!(目的是向下兼容)

3. 泛型方法:具有一个或多个类型变量的方法,称之为泛型方法!


class A<T> {
  public T fun(T t1) {}
}


fun()方法不是泛型方法!它是泛型类中的一个方法!


public <T> T fun(T t1) {} --> 它是泛型方法


* 泛型方法与泛型类没什么关系,泛型方法不一定非要在泛型类中!

4. 泛型在类中或方法中的使用

  * 泛型类中使用泛型:
    > 成员类型
    > 返回值和参数类型
    > 局部变量的引用上


class A<T> {
  private T bean; //泛型可在成员变量上使用
  public T fun(T t) {}/ /泛型可以在类中的方法上(返回值和参数类型)使用!


  public void fun2() { //泛型还可以在局部变量的引用类型上使用
    T b = ...
    new T(); //不行的!!!!!
  }
}


5. 泛型的继承和实现



class A<T> {}


class AA extends A<String> {} //不是泛型类,只是它爸爸是泛型类!

5.1 继承泛型类

  * 子类不是泛型类:需要给父类传递类型常量
    > 例如:当给父类传递的类型常量为String时,那么在父类中所有T都会被String替换!
  * 子类是泛型类:可以给父类传递类型常量,也可以传递类型变量


class AA1 extends A<Integer> {}
class AA2 extends A<T>{} //错误,不可行!!!
class AA3<E> extends A<E> {}



泛型的通配符



1. 通配符使用的场景



  方法的形参!


2. 通配符的优点

  使方法更加通用!


3. 通配符分类

  无界通配:?
  子类限定:? extends Object
  父类限定:? super Integer


4. 通配符缺点

  使变量使用上不再方便
  无界:参数和返回值为泛型的方法,不能使用!
  子类:参数为泛型的方法不能使用
  父类:返回值为泛型的方法不能使用


5. 比较通配符

boolean addAll(Collection<E> c)


List<Number> numList = new ArrayList<Number>();
List<Integer> intList = new ArrayList<Integer>();
numList.addAll(intList); //addAll(Collection<Number> c), 传递的是List<Integer>,编译失败




boolean addAll(Collection<? extends E> c)


List<Number> numList = new ArrayList<Number>();
List<Integer> intList = new ArrayList<Integer>();
numList.addAll(intList); //addAll(Collection<? extends Number> c), 传递的是List<Integer>,编译通过

public class Demo {
	@Test
	public void fun1() {
		Object[] objs = new Object[10];
		List list = new ArrayList();
		
		String[] strs = new String[10];
		List<String> strList = new ArrayList<String>();
		
		Object[] objArray = new String[10];
		objArray[0] = new Integer(100);//ArrayStoreException
//		List<Object> objList = new ArrayList<String>(); //泛型擦除后变成List objList = new ArrayList();
//		objList.add(new Integer(100));//泛型擦除后,可以在虚拟机中执行,所以编译器要拦截,出现编译错误
		/*
		 * 泛型引用和创建两端,给出的泛型变量必须相同!
		 */
	}
	
	public void fun2() {
		List<Integer> integerList = new ArrayList<Integer>();
		print(integerList);
		
		List<String> stringList = new ArrayList<String>();
		print(stringList);
	}
	
	/*
	 * 其中的?就是通配符
	 * 通配符只能出现在左边!即不能在new时使用通配符!!!
	 * List<?> list = new ArrayList<String>();
	 */
	/*
	 * ?它表示一个不确定的类型,它的值会在调用时确定下来
	 */
	public void print(List<?> list) {
		/*
		 * 当使用通配符时,对泛型类中的参数为泛型的方法起到了副作用,不能再使用!
		 */
//		list.add("hello");//add方法的参数为泛型
		/*
		 * 当使用通配符时,泛型类中返回值为泛型的方法,也作废了!
		 */
		Object s = list.get(0);
		/*
		 * 通配符好处:可以使泛型类型更加通用!尤其是在方法调用时形参使用通配符!
		 */
	}
	
	public void fun3() {
		List<Integer> intList = new ArrayList<Integer>();
		print1(intList);
		
		List<Long> longList = new ArrayList<Long>();
		print1(longList);
	}
	
	/*
	 * 给通配符添加了限定:
	 *   只能传递Number或其子类型
	 *   子类通配符对通用性产生了影响,但使用形参更加灵活
	 */
	public void print1(List<? extends Number> list) {
		/*
		 * 参数为泛型的方法还是不能使用
		 */
//		list.add(new Integer(100));
		/*
		 * 返回值为泛型的方法可用了!
		 */
		Number number = list.get(0);
	}
	
	public void fun4() {
		List<Integer> intList = new ArrayList<Integer>();
		print2(intList);
		
		List<Number> numberList = new ArrayList<Number>();
		print2(numberList);
		
		List<Object> objList = new ArrayList<Object>();
		print2(objList);
	}
	
	/*
	 * 给通配符添加了限定
	 *   只能传递Integer类型,或其父类型
	 */
	public void print2(List<? super Integer> list) {
		/*
		 * 参数为泛型的方法可以使用了
		 */
		list.add(new Integer(100));
		/*
		 * 返回值为泛型的方法,还是不能使用
		 */
		Object obj =  list.get(0);
	}
}





























































































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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值