java泛型之一

转载泛型:工作原理及其重要性
作者:Josh Juneau

代码github地址

泛型是什么?

考虑以下场景:您希望开发一个用于在应用中传递对象的容器。但对象类型并不总是相同。因此,需要开发一个能够存储各种类型对象的容器。

鉴于这种情况,要实现此目标,显然最好的办法是开发一个能够存储和检索 Object 类型本身的容器,然后在将该对象用于各种类型时进行类型转换。code 1 中的类演示了如何开发此类容器。

public class ObjectContainer {
    private Object obj;
    /**
     * @return the obj
     */
    public Object getObj() {
        return obj;
    }

    /**
     * @param obj the obj to set
     */
    public void setObj(Object obj) {
        this.obj = obj;
    }
}
//code 1

虽然这个容器会达到预期效果,但就我们的目的而言,它并不是最合适的解决方案。它不是类型安全的,并且要求在检索封装对象时使用显式类型转换,因此有可能引发异常。
code 2 中的代码演示如何使用该容器存储和检索值。

ObjectContainer myObj = new ObjectContainer();

// store a string
myObj.setObj("Test");
System.out.println("Value of myObj:" + myObj.getObj());
// store an int (which is autoboxed to an Integer object)
myObj.setObj(3);
System.out.println("Value of myObj:" + myObj.getObj());

List objectList = new ArrayList();
objectList.add(myObj);
// We have to cast and must cast the correct type to avoid ClassCastException!
String myStr = (String) ((ObjectContainer)objectList.get(0)).getObj(); 
System.out.println("myStr: " + myStr);
//code 2

可以使用泛型开发一个更好的解决方案,在实例化时为所使用的容器分配一个类型,也称泛型类型,这样就可以创建一个对象来存储所分配类型的对象。泛型类型是一种类型参数化的类或接口,这意味着可以通过执行泛型类型调用 分配一个类型,将用分配的具体类型替换泛型类型。然后,所分配的类型将用于限制容器内使用的值,这样就无需进行类型转换,还可以在编译时提供更强的类型检查。

code 3 中的类演示了如何创建与先前创建的容器相同的容器,但这次使用泛型类型参数,而不是 Object 类型。
public class GenericContainer<T> {
private T obj;

    public GenericContainer(){
    }

    // Pass type in as parameter to constructor
    public GenericContainer(T t){
        obj = t;
    }

    /**
     * @return the obj
     */
    public T getObj() {
        return obj;
    }

    /**
     * @param t the obj to set
     */
    public void setObj(T t) {
        obj = t;
    }
}
//code 3

最显著的差异是类定义包含 <T>,类字段 obj 不再是 Object 类型,而是泛型类型 T。类定义中的尖括号之间是类型参数部分,介绍类中将要使用的类型参数(或多个参数)。T 是与此类中定义的泛型类型关联的参数。

要使用泛型容器,必须在实例化时使用尖括号表示法指定容器类型。因此,以下代码将实例化一个 Integer 类型的 GenericContainer,并将其分配给 myInt 字段。

GenericContainer<Integer> myInt =  new GenericContainer<Integer>();

如果我们尝试在已经实例化的容器中存储其他类型的对象,代码将无法编译:

myInt.setObj(3);  // OK
myInt.setObj("Int"); // Won't Compile

使用泛型的好处
上面的示例已经演示了使用泛型的一些好处。一个最重要的好处是更强的类型检查,因为避开运行时可能引发的 ClassCastException 可以节省时间。

另一个好处是消除了类型转换,这意味着可以用更少的代码,因为编译器确切知道集合中存储的是何种类型。例如,在清单 4 所示代码中,我们来看看将 Object 容器实例存储到集合中与存储 GenericContainer 实例之间的差异。

List myObjList = new ArrayList();

// Store instances of ObjectContainer
for(int x=0; x <=10; x++){
    ObjectContainer myObj = new ObjectContainer();
    myObj.setObj("Test" + x);
    myObjList.add(myObj);
}
// Get the objects we need to cast
for(int x=0; x <= myObjList.size()-1; x++){
    ObjectContainer obj = (ObjectContainer) myObjList.get(x); 
    System.out.println("Object Value: " + obj.getObj());
}

List<GenericContainer> genericList = new ArrayList<GenericContainer>();

// Store instances of GenericContainer
for(int x=0; x <=10; x++){
    GenericContainer<String> myGeneric = new GenericContainer<String>();
    myGeneric.setObj(" Generic Test" + x);
    genericList.add(myGeneric);
}
// Get the objects; no need to cast to String

for(GenericContainer<String> obj:genericList){
    String objectString = obj.getObj();
    // Do something with the string...here we will print it
    System.out.println(objectString);
}
//code 4

注意,使用 ArrayList 时,我们可以使用括号表示法 (<GenericContainer>) 在创建时指定集合类型,指明我们将存储 GenericContainer 实例。该集合将只能存储 GenericContainer 实例(或 GenericContainer 的子类),无需在从集合检索对象时使用显式类型转换。

将泛型与 Collections API 结合使用的概念让我们能获得泛型提供的另外一个好处:允许开发可根据手头的任务定制的泛型算法。Collections API 本身是使用泛型开发的,如果不使用,Collections API 将永远无法容纳参数化类型。
分析泛型
以下各节将探讨泛型的更多特性。

如何使用泛型?##

泛型有许多不同用例。本文的第一个示例介绍了生成泛型对象类型的用例。这对于在类和接口层面了解泛型语法是个很好的起点。研究下代码,类签名包含一个类型参数部分,包括在类名后的尖括号 (< >) 内,例如:

public class GenericContainer<T> {
    ...
}

类型参数(又称类型变量)用作占位符,指示在运行时为类分配类型。根据需要,可能有一个或多个类型参数,并且可以用于整个类。根据惯例,类型参数是单个大写字母,该字母用于指示所定义的参数类型。下面列出每个用例的标准类型参数:

  • E:元素
  • K:键
  • N:数字
  • T:类型
  • V:值

S、U、V 等:多参数情况中的第 2、3、4 个类型
在上面的示例中,T 指示将分配的类型,因此可在实例化时为 GenericContainer 分配任何有效类型。注意,T 参数用于整个类,指示实例化时指定的类型。使用下面这行代码实例化对象时,将用 String 类型替换所有 T 参数:

GenericContainer<String> stringContainer = new GenericContainer<String>();

泛型也可用于构造函数中,传递类域初始化所需的类型参数。GenericContainer 的构造函数允许在实例化时传递任意类型:

GenericContainer gc1 = new GenericContainer(3);
GenericContainer gc2 = new GenericContainer("Hello");

注意,未分配类型的泛型称为原始类型。例如,要创建原始类型的 GenericContainer,可以使用以下代码:

GenericContainer rawContainer = new GenericContainer();

原始类型有时对于实现向后兼容很有用,但并不适用于日常代码。原始类型在编译时无需执行类型检查,导致代码在运行时易于出错

多种泛型类型##

有时,能够在类或接口中使用多种泛型类型很有帮助。通过在尖括号之间放置一个逗号分隔的类型列表,可在类或接口中使用多个类型参数。code 5 中的类使用一个接受以下两种类型的类演示了此概念:T 和 S。

如果我们回顾上一节中列出的标准类型命名约定,T 是第一种类型的标准标识符,S 是第二种类型的标准标识符。使用这两种类型生成一个使用泛型存储多个值的容器。

public class MultiGenericContainer<T, S> {
    private T firstPosition;
    private S secondPosition;
   
    public MultiGenericContainer(T firstPosition, S secondPosition){
        this.firstPosition = firstPosition;
        this.secondPosition = secondPosition;
    }
    
    public T getFirstPosition(){
        return firstPosition;
    }
    
    public void setFirstPosition(T firstPosition){
        this.firstPosition = firstPosition;
    }
    
    public S getSecondPosition(){
        return secondPosition;
    }
    
    public void setSecondPosition(S secondPosition){
        this.secondPosition = secondPosition;
    }
    
}
//code 5

MultiGenericContainer 类可用于存储两个不同对象,每个对象的类型可在实例化时指定。容器的用法如code 6 所示。

MultiGenericContainer<String, String> mondayWeather =new MultiGenericContainer<String, String>("Monday", "Sunny");
MultiGenericContainer<Integer, Double> dayOfWeekDegrees = new MultiGenericContainer<Integer, Double>(1, 78.0);

String mondayForecast = mondayWeather.getFirstPosition();
// The Double type is unboxed--to double, in this case. More on this in next section!
double sundayDegrees = dayOfWeekDegrees.getSecondPosition();
//code 6

类型推断和尖括号运算符
如前所述,泛型无需进行类型转换。例如,使用清单 5 中所示的 MultiGenericContainer 示例,如果调用 getFirstPosition() 或 getSecondPosition(),用于存储结果的字段必须与容器中该位置存储的对象的类型相同。

code 7 所示的示例中,我们看到实例化时分配给该容器的类型在检索值时无需进行类型转换。

MultiGenericContainer<String, String> mondayWeather = new MultiGenericContainer<String, String>("Monday", "Sunny");
MultiGenericContainer<Integer, Double> dayOfWeekDegrees = new MultiGenericContainer<Integer, Double>(1, 78.0);
String mondayForecast = mondayWeather.getFirstPosition(); // Works fine with String
// The following generates "Incompatible types" error and won't compile
int mondayOutlook = mondayWeather.getSecondPosition(); 
double sundayDegrees = dayOfWeekDegrees.getSecondPosition(); // Unboxing occurs
//code 7

考虑清单 7 中的第三行代码,由于 getSecondPosition() 的结果存储到 double 类型的字段中,因此无需进行类型转换。MultiGenericContainer 是用 MultiGenericContainer<String, Double> 实例化的,这怎么可能呢?借助将引用类型自动转换为原始类型的拆箱 操作,即可实现。同样,通过构造函数存储值时,使用自动装箱 操作将原始类型的 double 值存储为 Double 引用类型。

注:无法将原始类型用于泛型;只能使用引用类型。自动装箱和拆箱操作能够在使用泛型对象时将值存储为原始类型并检索原始类型的值。

类型引用可以在分配 getFirstPosition() 或 getSecondPosition() 调用结果时避免显式类型转换。根据 Oracle 文档,类型引用 是 Java 编译器的一项功能,可查看每种方法调用和对应的声明,从而确定支持调用的类型参数。换言之,编译器根据对象实例化过程中分配的类型确定可以使用的类型,在本例中,为 <String, String> 和 <Integer, Double>。引用算法尝试找到适用于所有参数的最特定的类型。

看看 MuliGenericContainer 的实例化,也可以使用类型引用避免重复类型声明。不必指定对象类型两次,只要编译器可以从上下文推断类型,即可以指定尖括号运算符 <>。因此,可以在实例化对象时使用尖括号运算符,如code 8 可见。

MultiGenericContainer<String, String> mondayWeather =new MultiGenericContainer<>("Monday", "Sunny");
MultiGenericContainer<Integer, Double> dayOfWeekDegrees = new MultiGenericContainer<>(1, 78.0);
//code 8

有界类型##

我们经常会遇到这种情况,需要指定泛型类型,但希望控制可以指定的类型,而非不加限制。有界类型 在类型参数部分指定 extendssuper 关键字,分别用上限下限限制类型,从而限制泛型类型的边界。例如,如果希望将某类型限制为特定类型或特定类型的子类型,请使用以下表示法:

<T extends UpperBoundType>

同样,如果希望将某个类型限制为特定类型或特定类型的超类型,请使用以下表示法:

<T super LowerBoundType>

code 9 的示例中,我们用先前使用的 GenericContainer 类,通过指定一个上限,将其泛型类型限制为 Number 或 Number 的子类。注意,GenericNumberContainer 这个新类指定泛型类型必须扩展 Number 类型。

public class GenericNumberContainer <T extends Number> {
    private T obj;

    public GenericNumberContainer(){
    }
    
    public GenericNumberContainer(T t){
        obj = t;
    }
    /**
     * @return
 the obj
     */
    public T getObj() {
        return obj;
    }

    /**
     * @param t the obj to set
     */
    public void setObj(T t) {
        obj = t;
    }
}
//code 9

该类可以很好地将其字段类型限制为 Number,但如果您尝试指定一个不在边界内的类型(如code 10 所示),将引发编译器错误。

GenericNumberContainer<Integer> gn = new GenericNumberContainer<Integer>();
gn.setObj(3);

// Type argument String is not within the upper bounds of type variable T
GenericNumberContainer<String> gn2 = new GenericNumberContainer<String>();
//code 10

泛型方法##

有时,我们可能不知道传入方法的参数类型。在方法级别应用泛型可以解决此类问题。方法参数可以包含泛型类型,方法也可以包含泛型返回类型。

假设我们要开发一个接受 Number 类型的计算器类。泛型可用于确保可将任何 Number 类型作为参数传递给此类的计算方法。例如,code 11 中的 add() 方法演示了如何使用泛型限制两个参数的类型,确保其包含 Number 的上限:

public static <N extends Number> double add(N a, N b){
    double sum = 0;
    sum = a.doubleValue() + b.doubleValue();
    return sum;
}   
//code 11

通过将类型限制为 Number,您可以将 Number 子类的任何对象作为参数传递。此外,通过将类型限制为 Number,我们还可以确保传递给该方法的任何参数将包含 doubleValue() 方法。要查看实际效果,如果您想添加一个 Integer 和一个 Float,可以按如下所示调用该方法:

double genericValue1 = Calculator.add(3, 3f);

通配符##

某些情况下,编写指定未知类型的代码很有用。问号 (?) 通配符可用于使用泛型代码表示未知类型。通配符可用于参数、字段、局部变量和返回类型但最好不要在返回类型中使用通配符,因为确切知道方法返回的类型更安全。

假设我们想编写一个方法来验证指定的 List 中是否存在指定的对象。我们希望该方法接受两个参数:一个是未知类型的 List,另一个是任意类型的对象。参见code 12

public static <T> void checkList(List<?> myList, T obj){
        if(myList.contains(obj)){
            System.out.println("The list contains the element: " + obj);
        } else {
            System.out.println("The list does not contain the element: " + obj);
        }
    }
//code 12

code 13 中的代码演示如何利用此方法。

// Create List of type Integer
List<Integer> intList = new ArrayList<Integer>();
intList.add(2);
intList.add(4);
intList.add(6);

// Create List of type String
List<String> strList = new ArrayList<String>();
strList.add("two");
strList.add("four");
strList.add("six");

// Create List of type Object
List<Object> objList = new ArrayList<Object>();
objList.add("two");
objList.add("four");
objList.add(strList);

checkList(intList, 3); 
// Output:  The list [2, 4, 6] does not contain the element: 3

checkList(objList, strList); 
/* Output:  The list [two, four, [two, four, six]] contains 
the element: [two, four, six] */

checkList(strList, objList);
/* Output:  The list [two, four, six] does not contain 
the element: [two, four, [two, four, six]] */
//code 13

有时要使用上限或下限限制通配符。与指定带边界的泛型类型极其相似,指定 extends 或 super 关键字加上通配符,后面跟用于上限或下限的类型,即可声明带边界的通配符类型。例如,如果我们要更改 checkList 方法使其只接受扩展 Number 类型的 List,可按code 14 所示编写代码。

public static <T> void checkNumber(List<? extends Number> myList, T obj){
    if(myList.contains(obj)){
        System.out.println("The list " + myList + " contains the element: " + obj);
    } else {
        System.out.println("The list " + myList + " does not contain the element: " + obj);
    }
}
//code 14
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,497评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,493评论 18 399
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,137评论 11 349
  • 四十不惑,四十的门槛就在眼前,而惑亦多。 一天的时间不经意就过去了,是日子过得好了吗? 一个孩子不声不响的拒绝回答...
    藏叶庭阅读 618评论 0 3
  • 沃尔夫问问
    MrHidden阅读 118评论 0 0