码迷,mamicode.com
首页 > 编程语言 > 详细

java泛型

时间:2014-09-26 00:08:57      阅读:347      评论:0      收藏:0      [点我收藏+]

标签:style   blog   http   color   os   使用   java   ar   for   

引言

泛型是指参数化类型的能力,可以定义泛型类型的类、接口或方法,随后编译器会用具体的类型来替换它

使用泛型的主要优点是:能够在编译时而不是在运行时检测错误

package java.lang; 
public interface Comparable { //JDK1.5之前
    public int compareTo(Object o);
}

package java.lang; 
public interface Comparable<T> { //JDK1.5之后
    public int compareTo(T o);
}

新泛型类型在编译时检测到可能的错误:

Comparable c = new Date();
System.out.println(c.compareTo("red"));

Comparable<Date> c = new Date();
System.out.println(c.compareTo("red"));

泛型类型必须是引用类型,不能用像int、double或char这样的基本类型来替换泛型类型:

ArrayList<int> intlist = new ArrayList<int>(); //error

必须使用:

ArrayList<Integer> intList = new ArrayList<Integer>();

java会自动打包操作:

ArrayList<Integer> intList = new ArrayList<Integer>();
intList.add(5); //java会自动地将5包装为new Integer(5)

定义泛型类和接口

下面看一个泛型类的例子:

bubuko.com,布布扣
public class GenericStack<E> {
    ArrayList<E> list = new ArrayList<E>();
    
    public int getSize() {
        return list.size();
    }
    
    public E peek() {
        return list.get(getSize() - 1);
    }
    
    public void push(E o) {
        list.add(o);
    }
    
    public E pop() {
        E o = list.get(getSize() - 1);
        list.remove(getSize() - 1);
        return o;
    }
    public boolean isEmpty() {
        return list.isEmpty();
    }
}
View Code

向栈中添加元素:

GenericStack<String> stack1 = new GenericStack<String>();
stack1.push("London");
stack1.push("Paris");
stack1.push("Berlin");
GenericStack<Integer> stack2 = new GenericStack<Integer>();
stack2.push(1);
stack2.push(2);
stack2.push(3);

定义泛型方法

public static void main(String args[])
{
    Integer[] integers = {1, 2, 3, 4, 5};
    String[] strings = {"London", "Paris", "New York", "Austin"};
    print(integers);
    print(strings);
}
    
public static <E> void print(E[] list) {
    for(int i = 0; i < list.length; i++)
        System.out.print(list[i] + " ");
        System.out.println();
}

通配泛型

下面的例子说明了为什么要使用通配泛型:

public class Main
{
    public static void main(String args[])
    {
        GenericStack<Integer> intStack = new GenericStack<Integer>();
        intStack.push(1);
        intStack.push(2);
        intStack.push(-2);
        System.out.print("The max number is " + max(intStack));
    }
    
    public static double max(GenericStack<Number> stack) {
        double max = stack.pop().doubleValue();
        
        while(!stack.isEmpty()) {
            double value = stack.pop().doubleValue();
            if(value > max)
                max = value;
        }
        return max;
    }
        
}

标记行会产生编译错误,因为intStack不是GenericStack<Number>的实例,尽管Integer是Number的子类型,但是GenericStack<Integer> 并不是 GenericStack<Number> 的子类型,为了避免这个问题,可以使用通配泛型类型 ,通配泛型类型有三种形式:

非受限通配:?  (它和? extends Object是一样的)

受限通配:? extends T  (表示T或T的一个未知子类型)

下限通配: ? super T   (表示T或T的一个未知父类型)

使用受限通配修改后的代码如下:

public class Main
{
    public static void main(String args[])
    {
        GenericStack<Integer> intStack = new GenericStack<Integer>();
        intStack.push(1);
        intStack.push(2);
        intStack.push(-2);
        System.out.print("The max number is " + max(intStack));
    }
    
    public static double max(GenericStack<? extends Number> stack) {
        double max = stack.pop().doubleValue();
        
        while(!stack.isEmpty()) {
            double value = stack.pop().doubleValue();
            if(value > max)
                max = value;
        }
        return max;
    }
        
}

再看下面的一个例子:

public class Main
{
    public static void main(String args[])
    {
        GenericStack<Integer> intStack = new GenericStack<Integer>();
        intStack.push(1);
        intStack.push(2);
        intStack.push(-2);
        print(intStack);
    }
    public static void print(GenericStack<?> stack) {
        while(!stack.isEmpty()) {
            System.out.print(stack.pop() + " ");
        }
    }    
}

什么时候使用下限通配?看下面的例子:

public class Main
{
    public static void main(String args[])
    {
        GenericStack<String> stack1 = new GenericStack<String>();
        GenericStack<Object> stack2 = new GenericStack<Object>();
        stack2.push("Java");
        stack2.push(2);
        stack1.push("Sun");
        add(stack1, stack2);
        print(stack2);
    }
    
    public static<T> void add(GenericStack<T> stack1,
            GenericStack<? super T> stack2) {
        while(!stack1.isEmpty()) {
            stack2.push(stack1.pop());
        }
    }
    
    public static void print(GenericStack<?> stack) {
        while(!stack.isEmpty()) {
            System.out.print(stack.pop() + " ");
        }
    }    
}

消除泛型和对泛型的限制

泛型是一种称为类型消除的方法实现的,泛型信息在运行时是不可用的,这种方法可以使泛型代码向后兼容使用原始类型的遗留代码

泛型存在于编译时。一旦编译器确认泛型类型是安全使用的,就将它转换为原始类型

例如下面的代码:

ArrayList<String> list = new ArrayList<String>();
list.add("Oklahoma");
String state = list.get(0);

被翻译成如下的代码的原始类型:

ArrayList list = new ArrayList();
list.add("Oklahoma");
String state = (String)list.get(0);

使用泛型类型是有一些限制的:

限制1、不能使用new E()

不能使用泛型类型参数创建实例: E object = new E()

限制2、不能使用new E()

不能使用泛型类型参数创建数组,如:E[] elements = new E[capacity];

可以通过创建一个Object类型的数组,然后将它的类型转化为E[]来规避这个限制,如:E[] elements = (E[]) new Object[capacity];

限制3、在静态环境中不允许类的参数是泛型类型

由于泛型类的所有实例都有相同的运行时类,所以泛型类的静态变量和方法是被它的所有实例所共享的,下面的代码非法:

public calss Test<E> {
    public  static Void m(E o1) {
    }
    public static E o1;
    static {
        E o2;
    }
}

限制4、异常类不能是泛型的

 

java泛型

标签:style   blog   http   color   os   使用   java   ar   for   

原文地址:http://www.cnblogs.com/bigjava/p/3901819.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!