欢迎使用CSDN-markdown编辑器

Java语言中泛型的起源

人总是懒惰的,总想着一劳永逸。程序员也不例外,还美其名曰:追求程序的通用性。这种追求是孜孜不倦的,前仆后继的。梳理一下人们所进行的努力吧:
1)第一阶段:在类和方法中,人们只能使用具体的类型:要么是基本类型,要么是自定义的类。这一阶段,程序的通用性很差。
2)第二阶段:在面向对象编程语言中,使用多态机制。例如,可以将方法的参数类型设为基类,那么该方法就可以接受从这个基类中导出的任何类作为参数。这样的方法更加通用一些,可应用的地方也多一些。但是,需要注意的是final类不能扩展。
3)第三阶段:有时候,拘泥于单继承体系,也会使程序受限太多。如果方法的参数是一个接口而不是一个类,这种限制就放松了很多。因为任何实现了该接口的类都满足该方法。这个阶段,使用了接口编程思想,让程序的通用性更强大了。
4)第四阶段:有时候,即使是使用了接口,对程序的约束也还是太强了,因为一旦指明了接口,它就要求你的代码必须使用特定的接口。而我们希望达到的目的是编写更通用的代码,要是代码应用于“某种不具体的类型”,而不是一个具体的接口或者类。这种“不具体的类型”,属于参数化类型,也就是说:把类型当做参数。当定义的时候,我们把类型当做一个参数;在使用的时候,我们才将这个参数用具体的类型来代替。为了实现参数化类型的目标,从而出现了泛型的机制。

什么是泛型编程

泛型的定义:泛型是JDK 1.5的一项新特性,它的本质是参数化类型(Parameterized Type)的应用,也就是说所操作的数据类型被指定为一个参数,在用到的时候再指定具体的类型。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口和泛型方法。泛型让编程人员能够使用类型抽象,通常用于集合里面。

泛型思想早在C++语言的模板(Templates)中就开始生根发芽,在Java语言处于还没有出现泛型的版本时,只能通过Object是所有类型的父类和类型强制转换两个特点的配合来实现类型泛化。例如在哈希表的存取中,JDK 1.5之前使用HashMap的get()方法,返回值就是一个Object对象,由于Java语言里面所有的类型都继承于java.lang.Object,那Object转型为任何对象成都是有可能的。但是也因为有无限的可能性,就只有程序员和运行期的虚拟机才知道这个Object到底是个什么类型的对象。在编译期间,编译器无法检查这个Object的强制转型是否成功,如果仅仅依赖程序员去保障这项操作的正确性,许多ClassCastException的风险就会被转嫁到程序运行期之中。

泛型技术在C#和Java之中的使用方式看似相同,但实现上却有着根本性的分歧,C#里面泛型无论在程序源码中、编译后的IL中(Intermediate Language,中间语言,这时候泛型是一个占位符)或是运行期的CLR中都是切实存在的,List与List就是两个不同的类型,它们在系统运行期生成,有自己的虚方法表和类型数据,这种实现称为类型膨胀,基于这种方法实现的泛型被称为真实泛型。Java语言中的泛型则不一样,它只在程序源码中存在,在编译后的字节码文件中,就已经被替换为原来的原始类型(Raw Type,也称为裸类型)了,并且在相应的地方插入了强制转型代码,因此对于运行期的Java语言来说,ArrayList与ArrayList就是同一个类。所以说泛型技术实际上是Java语言的一颗语法糖,Java语言中的泛型实现方法称为类型擦除,基于这种方法实现的泛型被称为伪泛型

泛型程序设计(Generic Programming)意味着编写的代码可以被很多不同类型的对象所重用。

下面是一个不用泛型例子,注意第3行代码,但这是让人很不爽的一点,因为程序员肯定知道自己存储在List里面的对象类型是Integer,但是在返回列表中元素时,还是必须强制转换类型,这是为什么呢?原因在于,编译器只能保证迭代器的 next()方法返回的是Object类型的对象,为保证Integer变量的类型安全,所以必须强制转换。这种转换不仅显得混乱,更可能导致类型转换异常ClassCastException,运行时异常往往让人难以检测到。保证列表中的元素为一个特定的数据类型,这样就可以取消类型转换,减少发生错误的机会,这也是泛型设计的初衷。

        List myIntList = new LinkedList();
        myIntList.add(new Integer(0));
        Integer x = (Integer)myIntList.iterator().next();

下面是一个使用了泛型的例子,在第1行代码中指定List中存储的对象类型为Integer,这样在获取列表中的对象时,不必强制转换类型了。

        List<Integer> myIntList = new LinkedList<Integer>();
        myIntList.add(new Integer(0));
        Integer x = myIntList.iterator().next();

泛型编程基本术语

以ArrayList和ArrayList做简要介绍:

  • 整个称为ArrayList泛型类型
  • ArrayList中的 E称为类型变量或者类型参数
  • 整个ArrayList 称为参数化的类型
  • ArrayList中的Integer称为类型参数的实例或者实际类型参数
  • ArrayList称为原始类型

定义简单的泛型

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值