Java 泛型(Generics)
Generics, 类似C++中的模版。
允许在定义类和接口的时候使用类型参数(type parameters), 声明的类型参数在使用的时候用具体的类型来替换。 如 ArrayList<String> files = new ArrayLis<String>();
可以使得程序有更好的可读性 和 安全性。
1.泛型类并没有自己独有的class类对象。
2.静态变量是被泛型类的所有实例共享的。
3。反省的类型参数不能用在java异常处理的catch语句中。
使用type parameters后,编译器会进行检查,避免插入错误类型的对象。
泛型类与一般的Java类基本相同,只是在类和接口定义上多出来了用<>声明的类型参数。一个类可以有多个类型参数,如 MyClass<X, Y, Z>。 每个类型参数在声明的时候可以指定上界。所声明的类型参数在Java类中可以像一般的类型一样作为方法的参数和返回值,或是作为域和局部变量的类型。但是由于类型擦除机制,类型参数并不能用来创建对象或是作为静态变量的类型。考虑下面的泛型类中的正确和错误的用法。
class ClassTest<X extends Number, Y, Z> {
    private X x;
    private static Y y; //编译错误,不能用在静态变量中
    public X getFirst() {
        //正确用法
        return x;
    }
    public void wrong() {
        Z z = new Z(); //编译错误,不能创建对象
    }
}  
最佳实践
在使用泛型的时候可以遵循一些基本的原则,从而避免一些常见的问题。
- 在代码中避免泛型类和原始类型的混用。比如List<String>和List不应该共同使用。这样会产生一些编译器警告和潜在的运行时异常。当需要利用JDK 5之前开发的遗留代码,而不得不这么做时,也尽可能的隔离相关的代码。
 - 在使用带通配符的泛型类的时候,需要明确通配符所代表的一组类型的概念。由于具体的类型是未知的,很多操作是不允许的。
 - 泛型类最好不要同数组一块使用。你只能创建new List<?>[10]这样的数组,无法创建new List<String>[10]这样的。这限制了数组的使用能力,而且会带来很多费解的问题。因此,当需要类似数组的功能时候,使用集合类即可。
 - 不要忽视编译器给出的警告信息。
 
普通泛型
- class Point< T>{ // 此处可以随便写标识符号,T是type的简称
 - private T var ; // var的类型由T指定,即:由外部指定
 - public T getVar(){ // 返回值的类型由外部决定
 - return var ;
 - }
 - public void setVar(T var){ // 设置的类型也由外部决定
 - this.var = var ;
 - }
 - };
 - public class GenericsDemo06{
 - public static void main(String args[]){
 - Point< String> p = new Point< String>() ; // 里面的var类型为String类型
 - p.setVar("it") ; // 设置字符串
 - System.out.println(p.getVar().length()) ; // 取得字符串的长度
 - }
 - };
 
----------------------------------------------------------
- class Notepad< K,V>{ // 此处指定了两个泛型类型
 - private K key ; // 此变量的类型由外部决定
 - private V value ; // 此变量的类型由外部决定
 - public K getKey(){
 - return this.key ;
 - }
 - public V getValue(){
 - return this.value ;
 - }
 - public void setKey(K key){
 - this.key = key ;
 - }
 - public void setValue(V value){
 - this.value = value ;
 - }
 - };
 - public class GenericsDemo09{
 - public static void main(String args[]){
 - Notepad< String,Integer> t = null ; // 定义两个泛型类型的对象
 - t = new Notepad< String,Integer>() ; // 里面的key为String,value为Integer
 - t.setKey("汤姆") ; // 设置第一个内容
 - t.setValue(20) ; // 设置第二个内容
 - System.out.print("姓名;" + t.getKey()) ; // 取得信息
 - System.out.print(",年龄;" + t.getValue()) ; // 取得信息
 - }
 - };
 
通配符
- class Info< T>{
 - private T var ; // 定义泛型变量
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public T getVar(){
 - return this.var ;
 - }
 - public String toString(){ // 直接打印
 - return this.var.toString() ;
 - }
 - };
 - public class GenericsDemo14{
 - public static void main(String args[]){
 - Info< String> i = new Info< String>() ; // 使用String为泛型类型
 - i.setVar("it") ; // 设置内容
 - fun(i) ;
 - }
 - public static void fun(Info< ?> temp){ // 可以接收任意的泛型对象
 - System.out.println("内容:" + temp) ;
 - }
 - };
 
受限泛型
- class Info< T>{
 - private T var ; // 定义泛型变量
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public T getVar(){
 - return this.var ;
 - }
 - public String toString(){ // 直接打印
 - return this.var.toString() ;
 - }
 - };
 - public class GenericsDemo17{
 - public static void main(String args[]){
 - Info< Integer> i1 = new Info< Integer>() ; // 声明Integer的泛型对象
 - Info< Float> i2 = new Info< Float>() ; // 声明Float的泛型对象
 - i1.setVar(30) ; // 设置整数,自动装箱
 - i2.setVar(30.1f) ; // 设置小数,自动装箱
 - fun(i1) ;
 - fun(i2) ;
 - }
 - public static void fun(Info< ? extends Number> temp){ // 只能接收Number及其Number的子类
 - System.out.print(temp + "、") ;
 - }
 - };
 
----------------------------------------------------------
- class Info< T>{
 - private T var ; // 定义泛型变量
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public T getVar(){
 - return this.var ;
 - }
 - public String toString(){ // 直接打印
 - return this.var.toString() ;
 - }
 - };
 - public class GenericsDemo21{
 - public static void main(String args[]){
 - Info< String> i1 = new Info< String>() ; // 声明String的泛型对象
 - Info< Object> i2 = new Info< Object>() ; // 声明Object的泛型对象
 - i1.setVar("hello") ;
 - i2.setVar(new Object()) ;
 - fun(i1) ;
 - fun(i2) ;
 - }
 - public static void fun(Info< ? super String> temp){ // 只能接收String或Object类型的泛型
 - System.out.print(temp + "、") ;
 - }
 - };
 
Java泛型无法向上转型
- class Info< T>{
 - private T var ; // 定义泛型变量
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public T getVar(){
 - return this.var ;
 - }
 - public String toString(){ // 直接打印
 - return this.var.toString() ;
 - }
 - };
 - public class GenericsDemo23{
 - public static void main(String args[]){
 - Info< String> i1 = new Info< String>() ; // 泛型类型为String
 - Info< Object> i2 = null ;
 - i2 = i1 ; //这句会出错 incompatible types
 - }
 - };
 
Java泛型接口
- interface Info< T>{ // 在接口上定义泛型
 - public T getVar() ; // 定义抽象方法,抽象方法的返回值就是泛型类型
 - }
 - class InfoImpl< T> implements Info< T>{ // 定义泛型接口的子类
 - private T var ; // 定义属性
 - public InfoImpl(T var){ // 通过构造方法设置属性内容
 - this.setVar(var) ;
 - }
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public T getVar(){
 - return this.var ;
 - }
 - };
 - public class GenericsDemo24{
 - public static void main(String arsg[]){
 - Info< String> i = null; // 声明接口对象
 - i = new InfoImpl< String>("汤姆") ; // 通过子类实例化对象
 - System.out.println("内容:" + i.getVar()) ;
 - }
 - };
 
----------------------------------------------------------
- interface Info< T>{ // 在接口上定义泛型
 - public T getVar() ; // 定义抽象方法,抽象方法的返回值就是泛型类型
 - }
 - class InfoImpl implements Info< String>{ // 定义泛型接口的子类
 - private String var ; // 定义属性
 - public InfoImpl(String var){ // 通过构造方法设置属性内容
 - this.setVar(var) ;
 - }
 - public void setVar(String var){
 - this.var = var ;
 - }
 - public String getVar(){
 - return this.var ;
 - }
 - };
 - public class GenericsDemo25{
 - public static void main(String arsg[]){
 - Info i = null; // 声明接口对象
 - i = new InfoImpl("汤姆") ; // 通过子类实例化对象
 - System.out.println("内容:" + i.getVar()) ;
 - }
 - };
 
Java泛型方法
- class Demo{
 - public < T> T fun(T t){ // 可以接收任意类型的数据
 - return t ; // 直接把参数返回
 - }
 - };
 - public class GenericsDemo26{
 - public static void main(String args[]){
 - Demo d = new Demo() ; // 实例化Demo对象
 - String str = d.fun("汤姆") ; // 传递字符串
 - int i = d.fun(30) ; // 传递数字,自动装箱
 - System.out.println(str) ; // 输出内容
 - System.out.println(i) ; // 输出内容
 - }
 - };
 
通过泛型方法返回泛型类型实例
- class Info< T extends Number>{ // 指定上限,只能是数字类型
 - private T var ; // 此类型由外部决定
 - public T getVar(){
 - return this.var ;
 - }
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public String toString(){ // 覆写Object类中的toString()方法
 - return this.var.toString() ;
 - }
 - };
 - public class GenericsDemo27{
 - public static void main(String args[]){
 - Info< Integer> i = fun(30) ;
 - System.out.println(i.getVar()) ;
 - }
 - public static < T extends Number> Info< T> fun(T param){//方法中传入或返回的泛型类型由调用方法时所设置的参数类型决定
 - Info< T> temp = new Info< T>() ; // 根据传入的数据类型实例化Info
 - temp.setVar(param) ; // 将传递的内容设置到Info对象的var属性之中
 - return temp ; // 返回实例化对象
 - }
 - };
 
使用泛型统一传入的参数类型
- class Info< T>{ // 指定上限,只能是数字类型
 - private T var ; // 此类型由外部决定
 - public T getVar(){
 - return this.var ;
 - }
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public String toString(){ // 覆写Object类中的toString()方法
 - return this.var.toString() ;
 - }
 - };
 - public class GenericsDemo28{
 - public static void main(String args[]){
 - Info< String> i1 = new Info< String>() ;
 - Info< String> i2 = new Info< String>() ;
 - i1.setVar("HELLO") ; // 设置内容
 - i2.setVar("汤姆") ; // 设置内容
 - add(i1,i2) ;
 - }
 - public static < T> void add(Info< T> i1,Info< T> i2){
 - System.out.println(i1.getVar() + " " + i2.getVar()) ;
 - }
 - };
 
Java泛型数组
- public class GenericsDemo30{
 - public static void main(String args[]){
 - Integer i[] = fun1(1,2,3,4,5,6) ; // 返回泛型数组
 - fun2(i) ;
 - }
 - public static < T> T[] fun1(T...arg){ // 接收可变参数
 - return arg ; // 返回泛型数组
 - }
 - public static < T> void fun2(T param[]){ // 输出
 - System.out.print("接收泛型数组:") ;
 - for(T t:param){
 - System.out.print(t + "、") ;
 - }
 - }
 - };
 
Java泛型的嵌套设置
- class Info< T,V>{ // 接收两个泛型类型
 - private T var ;
 - private V value ;
 - public Info(T var,V value){
 - this.setVar(var) ;
 - this.setValue(value) ;
 - }
 - public void setVar(T var){
 - this.var = var ;
 - }
 - public void setValue(V value){
 - this.value = value ;
 - }
 - public T getVar(){
 - return this.var ;
 - }
 - public V getValue(){
 - return this.value ;
 - }
 - };
 - class Demo< S>{
 - private S info ;
 - public Demo(S info){
 - this.setInfo(info) ;
 - }
 - public void setInfo(S info){
 - this.info = info ;
 - }
 - public S getInfo(){
 - return this.info ;
 - }
 - };
 - public class GenericsDemo31{
 - public static void main(String args[]){
 - Demo< Info< String,Integer>> d = null ; // 将Info作为Demo的泛型类型
 - Info< String,Integer> i = null ; // Info指定两个泛型类型
 - i = new Info< String,Integer>("汤姆",30) ; // 实例化Info对象
 - d = new Demo< Info< String,Integer>>(i) ; // 在Demo类中设置Info类的对象
 - System.out.println("内容一:" + d.getInfo().getVar()) ;
 - System.out.println("内容二:" + d.getInfo().getValue()) ;
 - }
 - };
 
Java 泛型(Generics)的更多相关文章
- Java 泛型(Generics) 综述
		
一. 引子 一般的类和方法.仅仅能使用详细类型:要么是基本类型.要么是自己定义类型.假设要编写能够应用于多种类型的代码,这样的刻板的限制对代码的束缚就会非常大. 多态算是一种泛化机制,但对代码的约束还 ...
 - 【Java心得总结三】Java泛型上——初识泛型
		
一.函数参数与泛型比较 泛型(generics),从字面的意思理解就是泛化的类型,即参数化类型.泛型的作用是什么,这里与函数参数做一个比较: 无参数的函数: public int[] newIntAr ...
 - Java深度历险(五)——Java泛型
		
作者 成富 发布于 2011年3月3日 | 注意:QCon全球软件开发大会(北京)2016年4月21-23日,了解更多详情!17 讨论 分享到:微博微信FacebookTwitter有道云笔记邮件 ...
 - Java泛型学习笔记 - (七)浅析泛型中通配符的使用
		
一.基本概念:在学习Java泛型的过程中, 通配符是较难理解的一部分. 主要有以下三类:1. 无边界的通配符(Unbounded Wildcards), 就是<?>, 比如List< ...
 - java 深度探险 java 泛型
		
Java泛型(generics)是JDK 5中引入的一个新特性,允许在定义类和接口的时候使用类型参数(type parameter).声明的类型参数在使用时用具体的类型来替换.泛型最主要的应用是在JD ...
 - Java泛型总结
		
1. 什么是泛型?泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类.可以把类型参数看作是使用参数化类型时指定的类型的 ...
 - Java泛型的好处
		
java 泛型是java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法. ...
 - java泛型的讲解
		
java泛型 什么是泛型? 泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类.可以把类型参数看作是使用参数化类型时指 ...
 - java 泛型深入之Set有用工具 各种集合泛型深入使用演示样例,匿名内部类、内部类应用于泛型探讨
		
java 泛型深入之Set有用工具 各种集合泛型深入使用演示样例,匿名内部类.内部类应用于泛型探讨 //Sets.java package org.rui.generics.set; import j ...
 
随机推荐
- POJ1056 IMMEDIATE DECODABILITY【数据结构】
			
题目地址:http://poj.org/problem?id=1056 Description An encoding of a set of symbols is said to be immedi ...
 - el表达式获取cookie
			
${cookie.name}将获得对应cookie的对象,比如我们用jsp将一段cookie发送给客户端. Cookie cookie = new Cookie("username" ...
 - javascript 数字字符串转为数字
			
var a="111"; comsole.log(a); //打印出来是个字符串类型 comsole.log(+a); //打印出来是个数字类型 comsole.log ...
 - isNotNull与isNotEmpty的区别
			
<isNotNull> 检查属性是否不为null <isNotEmpty> 检查Collection.size()的值,属性的String或String.valueOf()值, ...
 - linux 关机方式
			
linux 关机命令: 1- init 0 关机. 具体详情接受可以 init --help 查询 如下: init [OPTIONS...] {COMMAND} Send control comm ...
 - [大牛翻译系列]Hadoop(4)MapReduce 连接:选择最佳连接策略
			
4.1.4 为你的数据选择最佳连接策略 已介绍的每个连接策略都有不同的优点和缺点.那么,怎么来判断哪个最适合待处理的数据? 图4.11给出了一个决策树.这个决策树是于论文<A Compariso ...
 - 分享:mysql 随机查询数据
			
在mysql中查询5条不重复的数据,使用以下: 1 SELECT * FROM `table` ORDER BY RAND() LIMIT 5 就可以了.但是真正测试一下才发现这样效率非常低.一个1 ...
 - php ftp文件上传函数--新手入门参考
			
在 php编程中,用ftp上传文件比较多见,这里分享个简单入门型的ftp上传实例. <?php /** * ftp上传文件 * 学习ftp函数的用法 */ // 定义变量 $local_file ...
 - CLR via C# 内存管理读书记
			
1. CLR 垃圾回收采用基于代的机制, 在一次垃圾回收中存活下来的对象被提升到另一代 2. 在确认对象是否垃圾时,从一组根开始,根包括静态字段,方法参数,局部变量等 3. 使用CriticalFin ...
 - Spark菜鸟学习营Day4 单元测试程序的编写
			
Spark菜鸟学习营Day4 单元测试程序的编写 Spark相比于传统代码是比较难以调试的,单元测试的编写是非常必要的. Step0:需求分析 在测试案例编写前,需完成需求分析工作,明确程序所有的输入 ...