我们先来试着理解一下Object类,学习Java的应该都知道Object是所有类的父类,注意:那么这就意味着它的范围非常广!首先记住这点,如果你的参数类型时Object,那么的参数类型将非常广!
《Thinking in Java》中说很多原因促成了泛型的出现,最引人注目的一个原因就是为了创造容器类。这个要怎么来理解呢?我的理解是,可以抛开这个为了创造容器类这个,而是回到泛型的目的是限定某种类型上来。
所以我们现在能小结一下Object和T很重要的两点区别就是:
- Object范围非常广,而T从一开始就会限定这个类型(包括它可以限定类型为Object)。
- Object由于它是所有类的父类,所以会强制类型转换,而T从一开始在编码时(注意是在写代码时)就限定了某种具体类型,所以它不用强制类型转换。(之所以要强调在写代码时是因为泛型在虚拟机中会被JVM擦除掉它的具体类型信息,这点可参考泛型,在这里不做引申)。
比如在jdk中的List类是个泛型类。现在我们制定它的类型是Object。
List<Object> list = new ArrayList<Object>(); int i = 0; String s = "s"; list.add(i); list.add(s);
List本身是个泛型类,现在我们指定它接收Object类型的参数,此时就可以放置任意类型的参数进去,而在取出来是就必须得进行强制类型转换成具体的类型。
现在我们如果将List指定接收String类型的参数,那么这个List就只能放置String类型,且取出来时就不用进行强制类型转换。
这点给我们带来的启示是,在编写类似List类的时候,一定要注意是否用泛型。一定要多写几个泛型类,多讨论多理解,不然还是可能会搅在一起。
接着是?,这个可能在用到反射需要获取Class类型时用到,它的解释就是:接收一个不确定的类型,有点和Object一样。我对它一个理解是,如果只用”?”那么它和Object是一样的,但是”?”有比Object稍微“高级”有点的用法,就是它能缩小一个不确定的范围,利用类似”? extends Test”,这就意味着只接收接收Test类的继承类,是不是比Object的范围缩小了?
//对于类 Object,T可以修饰类,?不能修饰类 class A1 { } class A2<T> { } class A3<?> { //编译器提示代码错误 }
//对于方法参数 T和Object类型的参数可调用的方法是一样的,<?>必须基于<T>或<Object>来使用,就是一个通配符,指代所有类型 public <T> T getT(T t) { t.getClass(); return (T)t; } public Object getO(Object o) { o.getClass(); return (Object)o; } public List<?> get(List<?> o) { int a = o.size(); return (List<?>) o; } public void test() { BB<?> bb = new BB<>(); } class BB<Object>{}
//对于方法返回值<T>和<?>可以更精确的限制类型 public <T extends String> T getT2(T t) { t.getClass(); return t; } public Object getO2(Object o) { return o; } public List<? extends String> get2(List<?> l) { int a = l.size(); return (List<String>) l; }
//方法返回值不能用<T>,可以用<?> <T>不是一个实际类,它起一个指代作用,代替某个类,就是占位符,说明这里需要指定一个类型。 Object是实际的类, <?>是一个通配符号,如下表示这个类不指定类型,也可以使用extends和super限制匹配 public A2<T> getT3() { //编译器提示代码错误 return null; } public A2<Object> getO3() { return null; } public A2<?> get3() { return null; }
//方法返回值中<T>和<?>都可以用指定父类的方式限定返回值类型 public <T extends String> T getT4() { return null; } public A2<? extends String> get4() { return null; } //以上两个方法效果等同这个方法,也就是声明父类作为返回值(这里假设有类继承了String) public String getO4() { return null; }
//方法返回值可以用<?>指定子类类型,不能用<T>;个人觉得意义不大,用super的地方应该用extends也一样 public <T super String> T getT5() { //编译器提示代码错误 return null; } public A2<? super String> get5() { return null; }
总结:
1、Object是所有类的父类,代表所有类;是占位符,表示需要一个类型;通配符,匹配所有类型。
2、相对于Object,泛型通过指定父类可以缩小限制范围,可以限定匹配的类型的父类或子类。
3、泛型可以基于用户指定的类型进行显式或隐式的类型转换,在编译时进行类型转换检查,相对于Object避免了类型转换异常。
4、的好处是定义一个引用变量,可以指向多个对象。
List<?> list = new ArrayList<>(); list = new ArrayList<String>(); list = new ArrayList<Integer>(); List<Object> list2 = new ArrayList<>(); list2 = new ArrayList<Object>(); √ list2 = new ArrayList<Integer>(); ×