`
standalone
  • 浏览: 606356 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Java的Generics的几点限制

    博客分类:
  • java
阅读更多
参见

http://docs.oracle.com/javase/tutorial/java/generics/restrictions.html

To use Java generics effectively, you must consider the following restrictions:

  • Cannot Instantiate Generic Types with Primitive Types
  • Cannot Create Instances of Type Parameters
  • Cannot Declare Static Fields Whose Types are Type Parameters
  • Cannot Use Casts or instanceof With Parameterized Types
  • Cannot Create Arrays of Parameterized Types
  • Cannot Create, Catch, or Throw Objects of Parameterized Types
  • Cannot Overload a Method Where the Formal Parameter Types of Each Overload Erase to the Same Raw Type


Cannot Instantiate Generic Types with Primitive Types

这个是说你不能用primitive type来做类型参数,就是说下面这个使用泛型的例子是不对的:
class Pair<K, V> {

    private K key;
    private V value;

    public Pair(K key, V value) {
        this.key = key;
        this.value = value;
    }

    // ...
}

Pair<int, char> p = new Pair<>(8, 'a');  // compile-time error



Cannot Create Instances of Type Parameters
这个是不能直接new T()来创建类型参数的实例。下面的例子里,在run time,有关T或者E的类型信息都找不到了,所以你必须用反射的技术来做到:
T instantiateElementType(List<T> arg)
{
  return new T(); //causes a compilation error
}

public static <E> void append(List<E> list, Class<E> cls) throws Exception {
    E elem = cls.newInstance();   // OK
    list.add(elem);
}

这里反射的技术就是传一个类作为参数。


Cannot Declare Static Fields Whose Types are Type Parameters

这个容易引起混乱的,看下面这个例子:
如果类的静态变量允许是类型参数:
public class MobileDevice<T> {
    private static T os;

    // ...
}
MobileDevice<Smartphone> phone = new MobileDevice<>();
MobileDevice<Pager> pager = new MobileDevice<>();
MobileDevice<TabletPC> pc = new MobileDevice<>();

那么os的实际类型是什么呢?乱套了肯定。所以Java禁止这样做。

Cannot Use Casts or instanceof with Parameterized Types
因为java在编译是做了type erasure,所以无法区分ArrayList<Integer>和ArrayList<String>,所以Java不允许如下code编译通过:
public static <E> void rtti(List<E> list) {
    if (list instanceof ArrayList<Integer>) {  // compile-time error
        // ...
    }
}


Cannot Create Arrays of Parameterized Types
也就是说下面这行code会编译错误。

List<Integer>[] arrayOfLists = new List<Integer>[2];  // compile error

为什么要禁止这个呢?先看不同类型的object插入到array里会怎么样:

Object[] strings = new String[2];
strings[0] = "hi";   // OK
strings[1] = 100;    // An ArrayStoreException is thrown.


换成generic list呢,假设下面第一行code能够编译通过,那么第三行code在运行时就无法抛出ArrayStoreException了。所以Java要禁掉这样写法。

Object[] stringLists = new List<String>[];  // compiler error, but pretend it's allowed
stringLists[0] = new ArrayList<String>();   // OK
stringLists[1] = new ArrayList<Integer>();  // An ArrayStoreException should be thrown,
                                            // but the runtime can't detect it.


Cannot Create, Catch, or Throw Objects of Parameterized Types
一个generic的class不能继承Throwable(间接或直接都不行), 不能catch类型参数的实例(但可以throw)。例如:

// Extends Throwable indirectly
class MathException<T> extends Exception { /* ... */ }    // compile-time error

// Extends Throwable directly
class QueueFullException<T> extends Throwable { /* ... */ // compile-time error


public static <T extends Exception, J> void execute(List<J> jobs) {
    try {
        for (J job : jobs)
            // ...
    } catch (T e) {   // compile-time error
        // ...
    }
}

这些都会编译错误。

Cannot Overload a Method Where the Formal Parameter Types of Each Overload Erase to the Same Raw Type
这个是说因为type erasure的原因,会导致像
public class Example {
    public void print(Set<String> strSet) { }
    public void print(Set<Integer> intSet) { }
}

两个print方法完全一样,所以也被禁止。

以上是java文档上列出来的,还有一种情况:
 public List<? extends Foo> getFoos()
  {
    List<? extends Foo> foos = new ArrayList<? extends Foo>();
    foos.add(new SubFoo());
    return foos;
  }

第三行会编译错误“Cannot instantiate the type ArrayList<? extends Foo>", 原因是如果这行ok,那么编译器就不知道是否add一个SubFoo是安全的,如果foos被赋值成ArrayList<AltFoo> 呢?记住,compiler要在编译期解决所有类型检查的,所以这些能引起混乱的它肯定要禁掉了。参见stackoverflow上面的回答
多说一点的是更详细的关于genercis的使用,参见
http://tutorials.jenkov.com/java-generics/wildcards.html
关于? extends A的解释是List<? extends A>是只知道是一个list,可以装A或者A的子类,所以你可以读取里面的内容并cast成A, 但是不能insert。
分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    Java Generics and Collections (Java泛型与集合)

    书中可能涵盖了以下几个关键点: 1. **泛型的基本用法**:如何声明和使用泛型类、泛型方法以及通配符(如)。 2. **泛型的边界**:如何限制类型参数的范围,如`List&lt;? extends Number&gt;`只允许Number或其子类。 3. *...

    JavaGenericsFAQ

    考虑以下几点: - 是否有多个不同的类型将使用相同的逻辑? - 类型之间的关系是什么? - API的设计是否足够灵活以适应未来的扩展? ##### 2. 混合使用泛型与非泛型Java 在实践中,可能会遇到需要同时使用泛型和非...

    Java泛型编程快速入门

    在使用泛型时需要注意以下几点: 1. **定义泛型类**:在定义泛型类时,需要在类名后面的尖括号 `和 `&gt;` 之间定义类型参数。例如 `class JavaGenerics, V&gt;`,这里的 `K` 和 `V` 代表类型而不是具体的值。 2. **...

    Implementing Generics from JDK 5.0

    使用泛型时需要注意以下几点: - **类型参数的约束**:可以通过`extends`或`super`关键字对类型参数进行限定。 - **类型推断**:编译器能够根据上下文自动推断出类型参数的类型,无需显式指定。 - **通配符的使用**...

    Java 1.5 Generics For converter to 1.4.2-开源

    在使用这样的转换工具时,需要注意以下几点: 1. 转换可能不完美:尽管工具声称正常工作,但自动转换可能无法处理所有情况,特别是在涉及到复杂泛型使用和边界类型时。因此,转换后必须对代码进行彻底的测试和审查...

    JAVA词汇表下载.docJAVA词汇表下载.doc

    以下是一些重要的 Java 知识点: 1. 抽象类(Abstract Class):抽象类是一种不能被实例化的类,它的存在是为了被其他类继承。抽象类可以包含方法和变量,这些内容是所有子类共有的。如果一个类声明为抽象的,那么...

    Java 高级特性.doc

    以下是针对标题和描述中提到的几个关键知识点的详细说明: 1. 静态导入(Static Import): 静态导入允许我们将类中的静态成员直接作为当前类的一部分来使用,而不必每次都指定类名。例如,在示例中,如果不使用`...

    Java1.5泛型指南中文版

    Java 1.5引入了泛型(Generics)的概念,这是一个重要的语言特性,它允许开发者在编译时期指定集合或其他数据结构中的元素类型,从而避免了运行时期的类型转换错误。通过使用泛型,开发者可以在编程阶段就确保类型的...

    良葛格Java JDK 5.0学习笔记ch02

    在这一章中,良葛格可能会涵盖以下几个关键知识点: 1. **自动装箱与拆箱**:Java 5.0引入了自动装箱和拆箱机制,使得基本数据类型与对应的包装类之间可以无缝转换。例如,`Integer i = 10;` 和 `int num = i;` ...

    SCJP notes

    - `Generics` 文件名可能表示这是关于Java泛型的更多资料,可能包括其原理、限制和最佳实践。 学习这些笔记可以帮助你全面了解Java编程基础,掌握SCJP考试的核心内容,并为实际开发工作打下坚实的基础。通过深入...

    Java面试宝典

    ### Java面试宝典知识点详解 #### 一、抽象(Abstraction) 抽象是面向对象编程中的一个核心概念,指的是从现实世界或具体问题中提取出通用的模式或特性,同时忽略那些与当前目标无关的细节。在Java编程中,抽象...

    java 混淆工具,不可逆 jocky 也许是最好的了

    譬如Generics、Enhanced for Loop以及 Autoboxing/Unboxing等。但另人遗憾的是,倘若利用这些新的语法开发应用,就意味着不能够在JDK 1.4上运行,而JDK 1.4毕竟是目前最为普及的VM版本。幸运是,Jocky的另一个特色...

    Java源码反射-auto-resolver-excel-source:利用java的反射、克隆、泛型、LRU缓存等技术实现动态解析Excel

    下面我们将详细探讨这个项目涉及的Java知识点以及它们的应用。 首先,**反射**是Java提供的一种机制,通过`java.lang.Class`类和相关的API(如`Class.forName()`, `getMethod()`, `getField()`等),可以在运行时...

    kotlin文档

    原因有以下几点: 1. 表达性:Kotlin具有创新的语言特性,例如对类型安全的构建器和委托属性的支持。这些特性可以帮助开发者构建强大且易用的抽象。 2. 可扩展性:Kotlin支持协程,这让它能够构建能够在适度硬件...

    jdk1.5中的范型

    在进行此类重构时,需要考虑以下几点: - **识别可泛型化的类或方法**:分析现有代码,找出哪些类或方法可以受益于泛型。 - **逐步添加泛型支持**:可以从简单的方法或类开始,逐步扩展到更复杂的部分。 - **确保...

    JDK1.5和Tomcat5.5.9.rar

    尽管这些版本相对较旧,但在升级前需要考虑以下几点: 1. **测试**:在生产环境中使用新版本之前,必须进行全面的兼容性和性能测试。 2. **文档**:查阅官方文档和社区论坛,了解已知问题和解决方案。 3. **维护**...

    泛型的反射分析代码可参考复杂未看懂

    泛型的使用有以下几个关键点: 1. 类型参数:如`&lt;T&gt;`,表示一个未知类型。 2. 泛型类:在类定义中使用类型参数,如`class MyList&lt;T&gt; {...}`。 3. 泛型方法:在方法签名中使用类型参数,如`public &lt;T&gt; void print(T ...

    jocky 混肴编译rar包(ant和插件俩个版本)

    譬如Generics、Enhanced for Loop以及 Autoboxing/Unboxing等。但另人遗憾的是,倘若利用这些新的语法开发应用,就意味着不能够在JDK 1.4上运行,而JDK 1.4毕竟是目前最为普及的VM版本。幸运是,Jocky的另一个特色...

    joc eclipse plugin

    譬如Generics、Enhanced for Loop以及 Autoboxing/Unboxing等。但另人遗憾的是,倘若利用这些新的语法开发应用,就意味着不能够在JDK 1.4上运行,而JDK 1.4毕竟是目前最为普及的VM版本。幸运是,Jocky的另一个特色...

Global site tag (gtag.js) - Google Analytics