泛型及泛型的使用2

本文详细介绍了Java中的泛型有界类型,包括如何限制类型参数必须是特定类或接口的子类型,以及如何使用上界和下界通配符。通过示例展示了如何在泛型类和方法中应用这些概念,并解释了它们在实际编程中的作用和限制,如求数组平均值的方法和泛型集合的操作。此外,还提到了泛型接口、泛型类的约束以及泛型方法的使用场景。
摘要由CSDN通过智能技术生成

有界类型

public class MyClass<T extends Number>{  }  含义是要求传入T的类型必须是Number或者Number的子类型

在实际应用中可能会需要对传递给类型参数的类型加以限制。比如需要创建一个泛型类,它包含了一个求数组平均值的方法。这个数组的类型可以是整型、浮点型,但显然不能是字符串类型或是其他非数值类型。

Java提供了有界类型bounded types。在指定一个类型参数时,可以指定一个上界,声明所有的实际类型都必须是这个超类的直接或间接子类。

class classname <T extends superclass>

//要求传入的泛型类型必须是Number抽象类的子类 
public class Generic<T extends Number>{
    public double sum(T... params) {
        double res=0;
        for(int i=0;i<params.length;i++)
        	res+=params[i].doubleValue(); //这里是Number中定义的方法
        return res;
    }
}
	//注意:这里的extends不表示继承,只是说明传入的类型必须是Number或者Number的子类型

调用方法:
Generic<Long> g=new Generic<>():
double sum=g.sum(1L,2L,3L,4L,5L,6L);
System.out.println(sum);

因为Long是Number的子类,但是换一种类型String则会报错

Generic<String> g=new Generic<>(); //报错的原因是String不是Number的子类型

另外调用方式
Generic<Number> g=new Generic<>();
double sum=g.sum(1,2,3,4,5);  //写法正确,因为Integer(1)也是Number类型的

接口和类都可以用来做上界。class Stats<T extends Comparable> 。
这里需要注意:针对上界接口使用的关键字仍然是extends而非implements。一个类型参数
可以有多个限界,如class Stats<T extends Comparable & Serializable>。限界类型
用&分隔,因为逗号用来分隔类型参数。在多个限界中,可以有多个接口,但最多只能有一个类。如果用一个类作为限界,它必须是限界列表中的第一个。 

import java.io.Serializable;
import java.util.Date;
public class Test2 {
	public static void main(String[] args) {
//		MyClass<String> mc=null;
//		MyClass<Date> mc2=null;
		MyClass<Integer> mc1=new MyClass<>();
		MyClass<Double> mc2=new MyClass<>();
		
		MyClass<Number> mc3=new MyClass<>();
		mc3.add(123);  //Integer是Number的子类型
		mc3.add(12.345);
		
		Number kk=mc3.get();
		
		A2<B2> aa=new A2<>();
		
		D2<G2,T2> dd=new D2<>();
		
	}
}

class MyClass<T extends Number>{
	private T id;
	public void add(T t) {
		this.id=t;
	}
	
	public T get() {
		return id;
	}
}

class A2<T extends Comparable<T>&Serializable>{
	
}

class B2 implements Comparable<B2>,Serializable{

	@Override
	public int compareTo(B2 o) {
		return 0;
	}
}

//如果上限类型为类,不是接口,则只能有一个;如果有多个,则至少一个是接口
class C2<T extends Date & Serializable> {

}

class E2 {
}

class F2 {
}

class G2 extends E2 {
}

class H2 extends F2 {
}
class T2{}
class D2<T extends E2, F2> {
	T id;
	String name;
}

class Y2<T extends E2,String>{
	
}

通配符参数

使用一般用于接收参数 
void doSomething(Stats<?> ob) 表示这个参数ob可以是任意的Stats类型。
其中?表示一个不确定的类型,它的值会在调用时确定下来。 

List<?> list = new ArrayList<String>();

注意不能写成    List<Object> list=new ArrayList<String>() 

public class Test{
    public void pp(List<?> list){
        list.add("123"); //语法报错
    }
    public static void main(String[] args){
        List<String> list=new ArrayList<>();
        list.add("bbbb");
        Test t=new Test();
        t.pp(list);
    }
}

这里的List<?>中的?表示可以传人任意的类型参数,其中List<?> list可以匹配任意参数化的类型,但是到底匹配的是什么类型,只有以后才知道,所以cols=new ArrayList<Integer>和

cols = new ArrayList<String>都可以, 但是cols.add("abc")或cols.add(new Date())都不行

public static void printCollection(Collection<?> collection){
	for(Object obj:collection)    
        System.out.println(obj);
   	//collection.add("abc");//报错,因为collection不知道未来匹配的一定是String类型
  	collection.size();//不报错,此方法与参数类型没有关系
  	collection=new HashSet<Date>();//这是可以的
}

泛型类

public class A<T,ID>{
  private ID id;
  private T name;
}

定义泛型接口

public interface IA<T>{
   public T pp();
}

定义泛型类上给泛型添加约束 

public class MyClass<T extends IA>{}

在泛型类或者泛型接口中不允许直接使用super

List<? super Integer> array = new ArrayList<Integer>();

Integer 可以认为是 Integer 的 "父类"

List<? super Integer> array = new ArrayList<Number>();

Number 是 Integer 的 父类

List<? super Integer> array = new ArrayList<Object>();

Object 是 Integer 的 父类

如果要求泛型实现两个或者两个以上的接口

public class MyClass<T extends 接口1 & 接口2>{}


要求传入的类型T必须同时实现了接口1和接口2 

如果要求泛型必须继承B类,同时实现IC接口
public class MyClass<T extends B & IC >{}


如果要求泛型继承一个类并且实现接口
public class MyClass<T extends 类型 & 接口>{}

<? extends T>只能用于方法返回,告诉编译器此返参的类型的最小继承边界为T,
T和T的父类都能接收,但是入参类型无法确定,只能接受null的传入


<? super T>只能用于限定方法入参,告诉编译器入参只能是T或其子类型,而返参只能用Object类接收

          - ' ? '既不能用于入参也不能用于返参
          - 频繁往外读取内容的,适合用上界extends
          - 经常往里插入的,适合用下界Super

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

public class Test1 {
	public static void main(String[] args) {
		List<String> list=new ArrayList<>();
		list.add("bbbb");
		
		Test1 t1=new Test1();
		t1.pp(list);
		
//		t1.bb(list);    //List<String>和List<Object>之间没有任何关系
		
		List<Object> alist=new ArrayList<>();
		alist.add("1234");
		alist.add(1234);
		t1.bb(alist);
	}
	
	public void pp(List<?> list) {
//		list.add("123");  //语法报错
		
//		Object aa="123";
//		list.add(aa);
		
		for(Object tmp:list) {
			System.out.println(tmp);
		}
	}
	
	public void bb(List<Object> list) {}
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值