zoukankan      html  css  js  c++  java
  • Java泛型解析(01):认识泛型

    Java泛型解析(01):认识泛型
    What
         Java从1.0版本号到如今的8。中间Java5中发生了一个非常重要的变化,那就是泛型机制的引入。Java5引入了泛型,主要还是为了满足在1999年指定的最早Java规范之中的一个。经过了5年左右的时间,专家组定义了一套泛型规范,实现后通过測试投入到使用。

    所以说泛型是Java5以后才有的,欲知详情,继续往下看。

    Why
         换个角度想。Java5引入泛型。必然是它能带来优点,否则牛气的Java专家project师就要遭到吐槽了。

    我们来吐槽一下没有泛型的程序是怎么写的。

    [code01]
               ArrayList al = new ArrayList();
               al.add("ysjian001");
               al.add(1);
               al.add(new Object());   
          这段代码看似功能强大。为什么呢?由于它似乎可以往集合加入各种类型的对象(int 类型会被装箱成 Integer对象类型),貌似一些老程序猿也倾向于这么去做,并且他们可以理直气壮的告诉我理由:我这么做想存什么就存什么!先不否定这样的说法,让我们继续,看看以下代码。
    [code02]
                // 获取值的时候必须进行强制转换。然后调用相应对象的方法
                String first = (String) al.get(0);
        往集合里面存值就是为了后期取出来用的。而不是System.out.println(first),这里就产生了一个强制转换的问题,而往往这样的类型的强制转换在编译器是同意通过的,而敲代码的人们会犯下无意间的错误,错误的进行了强制转换,导致程序执行失败。

         强制类型转换导致的程序执行失败的原因是没有在编译器对类型进行控制。看看code01调用ArrayList对象的add方法,不论什么类型都是能够加入进行的,编译器无法进行错误检验。埋下了安全隐患,比如:
    [code03]
              ArrayList al = new ArrayList();
              // 无法进行错误检查,File对象能够加入进去,编译器和执行期都能够通过
              al.add(new File()); 
              String first = (String) al.get(0);  // 类型转换失败导致执行失败
    没有泛型的程序面临两个问题:
         1.编译器无法进行类型检查,能够向集合中加入随意类型的对象。
         2.取值时类型转换失败导致程序执行失败。

    没有泛型的程序导致的后果:
         1.程序的可读性有所减少。由于程序猿能够不受限制往集合中加入随意对象。
         2.程序的安全性遭到质疑,类型转换失败将导致程序执行失败。


         Java5泛型提供了一个更好的解决方式:类型參数(type parameters)。使用泛型的程序改善上述代码例如以下:
    [code04]
              ArrayList<String> al = new ArrayList<String>();
              al.add( "ysjian001");
              // al.add(new Thread()); // 定义了String类型參数,加入File对象会报错
              String first =  al.get(0);// 使用泛型后取值不用进行类型转换
         问:到这里。通过前后对照。泛型的优点是不是非常清楚了呢?为什么用泛型呢?
         答:由于出现编译错误比类在执行时出现强制类型转换异常要好得多,泛型的优点在于提高了程序的可读性和安全性。这也值程序设计的宗旨。

    Who
         使用泛型类是一件非常轻松的事,集合框架中的类都是泛型类。用起来非常方便。

    有人会想类型限制我们为什么不直接用数组呢?这个问题就好像问为什么集合优于数组。数组是固定的,而集合是能够自己主动扩展的。

    另外在实际中。实现一个泛型事实上并非那么easy。看一个员工和经理继承结构:

    [code05]
              public class Employee {
                    //......
              }
              public class Manager extends Employee {
                    // ......
              }
    当我们创建并使用一个员工的集合的时候,使用起来并不复杂:
    [code06]
              ArrayList<Employee> employees = new ArrayList<Employee>();
              employees.add(new Employee());       // 能够加入员工
              employees.add( new Manager());         // 能够加入经理,由于经理也是员工
         上述的使用毫无问题的,由于Manager is a Employee。典型的继承关系。可是当反过来的时候。可能不那么顺利了,比方:
    [code07]
              ArrayList<Manager> employees = new ArrayList<Manager>();
              employees.add(new Manager());          // 加入经理是正常的操作
              // employees.add(new Employee());     // 此时不能够加入Employee
    上面的代码就有问题了。而这样的需求又不是不存在,那么怎么办呢?不要着急,聪明的Java设计者发明了一个独具创新的新概念,通配符类型(wildcard type)。这里仅仅须要知道这个概念。后面会具体解说。

         大多数应用程序猿对泛型的熟练程度只停留在使用泛型上。像集合类中的List、Set和Map这些泛型集合用的非常多,他们不必考虑这些泛型集合的工作方式和原理。

    那么当把不同的泛型类混合在一起使用时,或者对Java5之前的遗留代码进行衔接时,可能会看到含糊不清的的错误消息。

    这样一来。程序猿就须要学习Java泛型来解决这个问题了,而不是在程序中胡乱推測了。终于,部分程序猿可能想要实现自己的泛型类和泛型方法。

         提炼出泛型程序设计的三种熟练程度就是:
         1.只使用泛型。
         2.学习泛型解决一些问题。
         3.掌握泛型,实现自己的泛型。

    How
         怎样使用泛型听起来是一件非常easy的事情,由于Sun公司的那些project师已经做了非常大努力。而需求总是会略微苛刻一点的。须要解决由于缺乏类型參数模糊不清的问题,或者我们有必要实现自己的泛型来满足业务需求,所以学习和掌握泛型是非常有必要的。
    泛型类:
         从简单的入手。定义一个泛型类:
    [code08]
         public class Couple<T> {
               private T wife ;
               private T husband ;
    
               public Couple(T wife, T husband) {
                        this.wife = wife;
                        this.husband = husband;
              }
               public void setWife(T wife) {this. wife = wife;}
               public void setHusband(T husband) {this. husband = husband;}
              
               public T getWife() {return wife;}
               public T getHusband() {return husband;}
         }
         Couple 夫妇类引入一个类型參数T。注意了,类型參数是用尖括号(< >)括起来的。而且放在类名后面,code08中的Couple类有一个类型參数。能够定义多个类型參数,格式为<T, K, V>, 类型參数能够用来定义方法的返回类型、參数类型、以及定义域或局部变量,如以下代码
    [code09]
              public class Couple<T, K, V> {......} // 多个类型參数用逗号隔开
              private T wife ;  // 类型參数定义域
              public T getWife() {return wife;}// 类型參数定义方法返回的类型
         在Java类库中,类型变量通经常使用大写的字母表示,E表示集合的元素类型。K和V分别表示映射表的keyword和值的类型,T(U或S)表示随意类型。
         一个简单的泛型类Couple定义好了,怎么使用呢?别着急。我们使用这个Couple类时指定一个详细的參数类型。如Person类:Couple<Person>
    [code10]
              Couple<Person>(Person,Person);
              setWife(Person);
              setHusband(Person);
              Person getWife();
              Person getHusband();
         code10中的代码是使用Person类型作为參数类型后。Couple类型的变化,注意这里     不是真正的变化成这样了。而是我们使用的时候这么理解,至于为什么呢?在后面会具体解说擦除。

    泛型方法:
         定义了泛型类有什么优点呢?通过前面的样例,这个泛型类能够让使用该类的用户在使用的时候指定才详细的类型。提高了一定的灵活性。那么看看泛型方法的定义是怎么回事?
    [code11]
         public class GenericMethod {
               public static <T> T getFirstValue(T[] values) {
                        return values[0];
              }
         }
         从已经学习的泛型类加上code11中的泛型方法的定义中总结一条,就是全部泛型都必须先以<T>(多个类型參数用逗号隔开。如<K, V>)的形式进行定义,这是必要的前提。回过头来看看上述泛型方法的定义,给方法定义了一个类型參数T,指定方法的返回值为T。方法的參数为T类型的数组。
         对方法的调用就比較直观了
    [code12]
              String[] values = { "JavaSE","CoreJava" ,"EffectiveJava"};
              String firstValue = GenericMethod.<String>getFirstValue(values);
         咋看调用还是有点复杂,为什么要在方法调用前用<String>呢?事实上这不是必要的。当我们将String[]类型的values传给方法时,编译器足以推断T的详细类型为String类型,所以<String>能够省略掉。

    [code13]
              String firstValue = GenericMethod.getFirstValue(values);
    总结:
         这一节里,对泛型有了一个总体的认识,知道它是什么?为什么要用它?谁会用它?以及怎样使用它?通过了泛型类和泛型方法的实践,感受了怎样实现自己的泛型。后面一节,将对泛型中通配符进行解说,以及虚拟机对泛型类和泛型方法的擦除。


    Java泛型解析(03):虚拟机运行泛型代码
    Java泛型解析(04):约束和局限性

    =====【感谢亲阅读寻常心的文章。亲若认为此文有帮助。顶一顶亲的支持将给我前进的动力】=====


  • 相关阅读:
    tablespaces
    转 房价
    Duplicate a whole line in Vim
    Dubbo+JStorm
    replace all
    ORACLE 最后表数据更新的时间
    list reverse
    python IDE
    string 方法
    java JIT AOT
  • 原文地址:https://www.cnblogs.com/gavanwanggw/p/6867241.html
Copyright © 2011-2022 走看看