Java泛型详解

简介:

一 概念

1.1 为什么需要泛型?

          当我们将一个对象放入集合中,集合不会记住此对象的类型,当再次从集合中取出此对象时,该对象的编译类型变成了Object类型,但其运行时类型任然为其本身类型。因此,取出集合元素时需要人为的强制类型转化到具体的目标类型,且很容易出现“java.lang.ClassCastException”异常。使用泛型就可以解决此类问题。

1.2 什么是泛型?

        泛型(Generic type 或者 generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类。可以把类型参数看作是使用参数化类型时指定的类型的一个占位符,就像方法的形式参数是运行时传递的值的占位符一样。         

        可以在集合框架(Collection framework)中看到泛型的动机。例如,Map 类允许您向一个 Map 添加任意类的对象,即使最常见的情况是在给定映射(map)中保存某个特定类型(比如 String)的对象。因为 Map.get() 被定义为返回 Object,所以一般必须将 Map.get() 的结果强制类型转换为期望的类型,如下面的代码所示:

Map m = new HashMap();
m.put("key", "blarg");
String s = (String) m.get("key");

要让程序通过编译,必须将 get() 的结果强制类型转换为 String,并且希望结果真的是一个 String。但是有可能某人已经在该映射中保存了不是 String 的东西,这样的话,上面的代码将会抛出 ClassCastException。

理想情况下,您可能会得出这样一个观点,即 m 是一个 Map,它将 String 键映射到 String 值。这可以让您消除代码中的强制类型转换,同时获得一个附加的类型检查层,该检查层可以防止有人将错误类型的键或值保存在集合中。这就是泛型所做的工作。

二 定义和使用

2.1 定义

        泛型参数的命名风格为:推荐你用简练的名字作为形式类型参数的名字(如果可能,单个字符)。最好避免小写字母,这使它和其他的普通的形式参数很容易被区分开来。使用T代表类型,无论何时都没有比这更具体的类型来区分它。这经常见于泛型方法。如果有多个类型参数,我们可能使用字母表中T的临近的字母,比如S。如果一个泛型函数在一个泛型类里面出现,最好避免在方法的类型参数和类的类型参数中使用同样的名字来避免混淆。对内部类也是同样。

例:

<span style="font-size:10px;"> <span style="font-size:12px;">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 ;  
    }  
};  </span></span>
2.2 使用方法

2.2.1 通配符(?)与上下界(extends ,super )

        使用泛型类的时候,既可以指定一个具体的类型,如List<String>就声明了具体的类型是String;也可以用通配符?来表示未知类型,如List<?>就声明了List中包含的元素类型是未知的。 通配符所代表的其实是一组类型,但具体的类型是未知的。List<?>所声明的就是所有类型都是可以的。但是List<?>并不等同于List<Object>。List<Object>实际上确定了List中包含的是Object及其子类,在使用的时候都可以通过Object来进行引用。而List<?>则其中所包含的元素类型是不确定。其中可能包含的是String,也可能是 Integer。如果它包含了String的话,往里面添加Integer类型的元素是错误的。正因为类型未知,就不能通过new ArrayList<?>()的方法来创建一个新的ArrayList对象。因为编译器无法知道具体的类型是什么。但是对于List<?>中的元素确总是可以用Object来引用的,因为虽然类型未知,但肯定是Object及其子类。

例:extends通配符,向上造型一个泛型对象的引用

?extends XX,XX 类是用来限定通配符的上界,XX 类是能匹配的最顶层的类,它只能匹配 XX 类以及 XX 类的子类

//原始版本
public void drawAll(List<Shape> shapes) {
     for (Shapes:shapes) 
		 {
           s.draw(this);
         }
    }
//使用边界通配符的版本
public void drawAll(List<?exends Shape> shapes) {
     for (Shapes:shapes)
		 {
         s.draw(this);
         }
    }
例: ? super通配符,向下造型一个泛型对象的引用
? super XX,XX 类是用来限定通配符的下界,XX 类是能匹配的最底层的类,它只能匹配 XX 类及子类。

List<Shape> shapes = new ArrayList<Shape>();
List<? super Cicle> cicleSupers = shapes;
cicleSupers.add(new Cicle()); //OK, subclass of Cicle also OK
cicleSupers.add(new Shape());//Error

2.2.2 泛型接口

 <span style="font-size:12px;">   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()) ;  
        }  
    };  </span>
2.2.3 泛型数组

    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 + "、") ;  
            }  
        }  
    };  







目录
相关文章
|
1月前
|
存储 安全 Java
JAVA泛型
JAVA泛型
12 0
|
2月前
|
安全 Java
玩转Java泛型
玩转Java泛型
21 0
|
8月前
|
Java 容器
JAVA泛型详解
JAVA泛型详解
|
10月前
|
Java
详解JAVA泛型
1.概述 泛型,自JDK1.5引入的语法糖,底层本质上就是类型强转,目的是为了“参数化类型”。所谓的参数化类型是希望参数的类型能动态指定。泛型泛型既可以修饰属性,又可以修饰方法。
77 0
|
安全 Java 程序员
java泛型总结
Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。 泛型的本质是参数化类型,即给类型指定一个参数,然后在使用时再指定此参数具体的值,那样这个类型就可以在使用时决定了。这种参数类型可以用在类、接口和方法中,分别被称为泛型类、泛型接口、泛型方法。
70 0
|
Java
Java泛型——特点
Java泛型——特点
83 0
|
Java
Java泛型详解2
Java泛型详解2
86 0
Java泛型详解2
|
存储 安全 Java
Java泛型详解1
Java泛型详解1
57 0
|
Java 容器
java泛型理解
集合容器类在设计阶段 / 声明阶段不能确定这个容器到底实际存的是什么类型的 对象,所以 在 JDK1.5 之前只能把元素类型设计为 Object , JDK1.5 之后使用泛型来 解决。
java泛型理解
|
Java
你必须知道的Java泛型(下)
前言 文本已收录至我的GitHub仓库,欢迎Star:github.com/bin39232820… 种一棵树最好的时间是十年前,其次是现在
124 0

热门文章

最新文章