泛型

1、泛型
在程序编码中一些包含类型参数的类型,也就是说泛型的参数只可以代表类,不能代表个别对象。
在程序编码中一些包含参数的类。其参数可以代表类或对象等等。(人们大多把这称作模板)不论使用哪个定义,泛型的参

数在真正使用泛型时都必须指明。
一些强类型编程语言支持泛型,其主要目的是加强类型安全及减少类转换的次数,但一些支持泛型的编程语言只能达到部分

目的。

2、java的泛型
java泛型的参数只可以代表类,不能代表个别对象。由于java泛型的类型参数之实际类型在编译时会被消除,所以无法在运

行时得知其类型参数的类型。java编译器在编译泛型时会自动加入类型转换的编码,故运行速度不会因为使用泛型而加快。

java允许对个别泛型的类型参数进行约束,包括以下两种形式(假设T是泛型的类型参数,C是一般类、泛类,或是泛型的类

型参数):T实现接口I。T是C,或继承自C。一个泛型类不能实现Throwable接口。

例子
List arrayList = new ArrayList();
arrayList.add(“aaaa”);
arrayList.add(100);

for(int i = 0; i< arrayList.size();i++){
String item = (String)arrayList.get(i);
Log.d(“泛型测试”,"item = " + item);
}

毫无疑问,程序的运行结果会以崩溃结束:

java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
ArrayList可以存放任意类型,例子中添加了一个String类型,添加了一个Integer类型,再使用时都以String的方式使用,

因此程序崩溃了。为了解决类似这样的问题(在编译阶段就可以解决),泛型应运而生。

我们将第一行声明初始化list的代码更改一下,编译器会在编译阶段就能够帮我们发现类似这样的问题。

List arrayList = new ArrayList();

//arrayList.add(100); 在编译阶段,编译器就会报错

3、泛型只在编译阶段有效。

List stringArrayList = new ArrayList();
List integerArrayList = new ArrayList();

Class classStringArrayList = stringArrayList.getClass();
Class classIntegerArrayList = integerArrayList.getClass();

if(classStringArrayList.equals(classIntegerArrayList)){
Log.d(“泛型测试”,“类型相同”);
}

输出结果:D/泛型测试: 类型相同。

通过上面的例子可以证明,在编译之后程序会采取去泛型化的措施。也就是说Java中的泛型,只在编译阶段有效。在编译过

程中,正确检验泛型结果后,会将泛型的相关信息擦出,并且在对象进入和离开方法的边界处添加类型检查和类型转换的方

法。也就是说,泛型信息不会进入到运行时阶段。

对此总结成一句话:泛型类型在逻辑上看以看成是多个不同的类型,实际上都是相同的基本类型。

4、泛型有三种使用方式,分别为:泛型类、泛型接口、泛型方法。

泛型类

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

类,如:List、Set、Map。

泛型类的最基本写法(这么看可能会有点晕,会在下面的例子中详解):

class 类名称 <泛型标识:可以随便写任意标识号,标识指定的泛型的类型>{
private 泛型标识 /(成员变量类型)/ var;

}
}
一个最普通的泛型类

//此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型
//在实例化泛型类时,必须指定T的具体类型
public class Generic{
//key这个成员变量的类型为T,T的类型由外部指定
private T key;

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

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

}

//泛型的类型参数只能是类类型(包括自定义类),不能是简单类型
//传入的实参类型需与泛型的类型参数类型相同,即为Integer.
Generic genericInteger = new Generic(123456);

//传入的实参类型需与泛型的类型参数类型相同,即为String.
Generic genericString = new Generic(“key_vlaue”);
Log.d(“泛型测试”,"key is " + genericInteger.getKey());
Log.d(“泛型测试”,"key is " + genericString.getKey());

12-27 09:20:04.432 13063-13063/? D/泛型测试: key is 123456
12-27 09:20:04.432 13063-13063/? D/泛型测试: key is key_vlaue

定义的泛型类,就一定要传入泛型类型实参么?并不是这样,在使用泛型的时候如果传入泛型实参,则会根据传入的泛型实

参做相应的限制,此时泛型才会起到本应起到的限制作用。如果不传入泛型类型实参的话,在泛型类中使用泛型的方法或成

员变量定义的类型可以为任何的类型。

Generic generic = new Generic(“111111”);
Generic generic1 = new Generic(4444);
Generic generic2 = new Generic(55.55);
Generic generic3 = new Generic(false);

Log.d(“泛型测试”,"key is " + generic.getKey());
Log.d(“泛型测试”,"key is " + generic1.getKey());
Log.d(“泛型测试”,"key is " + generic2.getKey());
Log.d(“泛型测试”,"key is " + generic3.getKey());

D/泛型测试: key is 111111
D/泛型测试: key is 4444
D/泛型测试: key is 55.55
D/泛型测试: key is false
注意:

泛型的类型参数只能是类类型,不能是简单类型。
不能对确切的泛型类型使用instanceof操作。如下面的操作是非法的,编译时会出错。
  if(ex_num instanceof Generic){ }

泛型接口

泛型接口与泛型类的定义及使用基本范围相同。泛型接口常被用在各种类的生产器中

//定义一个泛型接口
public interface Generator {
public T next();
}

当实现泛型接口的类,未传入泛型实参时:

/**

  • 未传入泛型实参时,与泛型类的定义相同,在声明类的时候,需将泛型的声明也一起加到类中
  • 即:class FruitGenerator implements Generator{
  • 如果不声明泛型,如:class FruitGenerator implements Generator,编译器会报错:“Unknown class”
    */
    class FruitGenerator implements Generator{
    @Override
    public T next() {
    return null;
    }
    }

当实现泛型接口的类,传入泛型实参时:

/**

  • 传入泛型实参时:

  • 定义一个生产器实现这个接口,虽然我们只创建了一个泛型接口Generator

  • 但是我们可以为T传入无数个实参,形成无数种类型的Generator接口。

  • 在实现类实现泛型接口时,如已将泛型类型传入实参类型,则所有使用泛型的地方都要替换成传入的实参类型

  • 即:Generator,public T next();中的的T都要替换成传入的String类型。
    */
    public class FruitGenerator implements Generator {

    private String[] fruits = new String[]{“Apple”, “Banana”, “Pear”};

    @Override
    public String next() {
    Random rand = new Random();
    return fruits[rand.nextInt(3)];
    }
    }

泛型通配符

类型通配符一般是使用?代替具体的类型实参,此处’?’是类型实参,而不是类型形参

泛型方法

/**

  • 泛型方法的基本介绍
  • @param tClass 传入的泛型实参
  • @return T 返回值为T类型
  • 说明:
  • 1)public 与 返回值中间<T>非常重要,可以理解为声明此方法为泛型方法。
    
  • 2)只有声明了<T>的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。
    
  • 3)<T>表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T。
    
  • 4)与泛型类的定义一样,此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型。
    

*/
public T genericMethod(Class tClass)throws InstantiationException ,
IllegalAccessException{
T instance = tClass.newInstance();
return instance;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值