Java 泛型概述

泛型的作用

第一是泛化。

可以用T代表任意类型Java语言中引入泛型是一个较大的功能增强不仅语言、类型系统和编译器有了较大的变化,以支持泛型,而且类库也进行了大翻修,所以许多重要的类,比如集合框架,都已经成为泛型化的了,这带来了很多好处。

第二是类型安全

泛型的一个主要目标就是提高Java程序的类型安全,使用泛型可以使编译器知道变量的类型限制,进而可以在更高程度上验证类型假设。如果不用泛型,则必须使用强制类型转换,而强制类型转换不安全,在运行期可能发生ClassCast Exception异常,如果使用泛型,则会在编译期就能发现该错误。

第三是消除强制类型转换。

泛型可以消除源代码中的许多强制类型转换,这样可以使代码更加可读,并减少出错的机会。

第四是向后兼容。

支持泛型的Java编译器(例如JDK1.5中的Javac)可以用来编译经过泛型扩充的Java程序(Generics Java程序),但是现有的没有使用泛型扩充的Java程序仍然可以用这些编译器来编译。

泛型的优点

1、类型安全

泛型的主要目标是提高Java程序的类型安全。通过知道使用泛型定义的变量的类型限制,编译器可以在非常高的层次上验证类型假设。没有泛型,这些假设就只存在于系统开发人员的头脑中。

通过在变量声明中捕获这一附加的类型信息,泛型允许编译器实施这些附加的类型约束。类型错误就可以在编译时被捕获了,而不是在运行时当作ClassCastException展示出来。将类型检查从运行时挪到编译时有助于Java开发人员更早、更容易地找到错误,并可提高程序的可靠性。

2、消除强制类型转换

泛型的一个附带好处是,消除源代码中的许多强制类型转换。这使得代码更加可读,并且减少了出错机会。尽管减少强制类型转换可以提高使用泛型类的代码的累赞程度,但是声明泛型变量时却会带来相应的累赞程度。在简单的程序中使用一次泛型变量不会降低代码累赞程度。但是对于多次使用泛型变量的大型程序来说,则可以累积起来降低累赞程度。所以泛型消除了强制类型转换之后,会使得代码加清晰和筒洁。

3、更高的运行效率

在非泛型编程中,将筒单类型作为Object传递时会引起Boxing(装箱)和Unboxing(拆箱)操作,这两个过程都是具有很大开销的。引入泛型后,就不必进行Boxing和Unboxing操作了,所以运行效率相对较高,特别在对集合操作非常频繁的系统中,这个特点带来的性能提升更加明显。

4、潜在的性能收益

泛型为较大的优化带来可能。在泛型的初始实现中,编译器将强制类型转换(没有泛型的话,Java系统开发人员会指定这些强制类型转换)插入生成的字节码中。但是更多类型信息可用于编译器这一事实,为未来版本的JVM的优化带来可能

泛型概述

Java SE 5.0以前操作集合的缺点:

当把一个对象存入集合后,集合会“忘记”这个对象的类型,将该对象从集合中取出时,这个对象的编译类型就变成了Object类型。

从集合中取出对象,需要执行类型转换操作:

由于没有类型检查,可以向集合添加任意对象,get方法在取出元素时,如果进行强制类型转换就很容易出错

ArrayList files = new ArrayList();
 . . . 
String filename = (String) files.get(0); 

泛型提供了参数化类型

在定义集合类时,可以使用“<参数化类型>”的方式指定该类中方法操作的数据类型。如

ArrayList<String> files = new ArrayList<String>(); 

限定了ArrayList集合只能存储String类型元素

取出对象的类型为String,无需执行类型转换。

没有使用泛型的Java代码警告

 泛型的使用

如何定义泛型List对象

List<T>

T是类型变量

T也可以是泛型变量类型

泛型经常被称为参数化类型,它能够像方法一样接受不同类型的参数

<Integer>存储IntegerList

        List <Integer> numbers = new ArrayList<Integer>();
        for (int i = 1; i <= 49; i++){
            numbers.add(i);
        }
Collections.shuffle(numbers);//对集合中的元素随机排序
        List<Integer> subnumbers=    numbers.subList(0, 6);
        Collections.sort(subnumbers);
        System.out.println(subnumbers);

泛型中使用通配符

泛型中可以使用“?”通配符作为参数,表示该泛型可以接收任意类型的数据

public static void print(List<?>  list) {
for (Object element : list) {
                     System.out.println(element );
               } }

上界通配符:

泛型中只允许一个类自身(接口)或者其子类(实现该接口的类)作为参数传入。

ArrayList<? extends 父类(接口)>

下界通配符:

泛型中只允许一个类自身或者该类的父类作为参数传入。

ArrayList<? super 子类>

自定义泛型

Point类具有两个成员,类型待定

使用private T x表示x的类型为参数T

public class GerDemo {
public static void main(String[] args) {
Point<Integer> p1=new Point<Integer>();  //点p1
p1.setX(5);
p1.setY(8);
System.out.println("p1:("+p1.getX()+","+p1.getY()+")");
Point<Integer> p2=new Point<Integer>(34,98); //点p2
System.out.println("p2:("+p2.getX()+","+p2.getY()+")");
Point<Double> p3=new Point<Double>(); //点p3
p3.setX(5.9);
p3.setY(8.0);
System.out.println("p3:("+p3.getX()+","+p3.getY()+")");
Point<Double> p4=new Point<Double>(2.1,9.2); //点p4
System.out.println("p4:("+p4.getX()+","+p4.getY()+")");
}
}
public class Point<T>
{  
    private T x;
    private T y;
    public Point() {}
    public Point(T x, T y) {  this.x = x;  this.y = y;  } 
    public T getX() {  return  x;  }
    public T getY() {  return  y;  }    
    public void setX(T x) {  this.x = x;  }
    public void setY(T y) {  this.y = y; }    
}
编写一个泛型方法,能够对数组求最大值和最小值(实现通用的求极值算法)。

class ArrayAlg {
   public static <T extends Comparable> Pair<T> minmax(T[ ] a) {
        if (a == null || a.length == 0) {
            return null;
        }
        T min = a[0];T max = a[0];
        for (int i = 1; i < a.length; i++) {
            if (min.compareTo(a[i]) > 0) {min = a[i];}
            if (max.compareTo(a[i]) < 0) { max = a[i];}
        }
        return new Pair<T>(min, max);
    }
}
使用泛型方法
创建一个日历的数组,求出最大和最小的日期
Pair<GregorianCalendar > mm 使用 GregorianCalendar Pair<T>
传递类型参数
Java.util.GregorianCalendar实现了接口 java.lang.Comparable <T>
        GregorianCalendar[ ] birthdays = {
            new GregorianCalendar(1906, Calendar.DECEMBER, 9), 
            new GregorianCalendar(1815, Calendar.DECEMBER, 10), 
            new GregorianCalendar(1903, Calendar.DECEMBER, 3), 
            new GregorianCalendar(1910, Calendar.JUNE, 22), 
        };
        Pair<GregorianCalendar> mm =   ArrayAlg.minmax(birthdays);
        System.out.println("min = " + mm.getFirst().getTime());
        System.out.println("max = " + mm.getSecond().getTime());

     

如何求出一个int类型数组的最大值和最小值?

如何求出一个String类型数组的最大值和最小值(字典顺序)

String[ ] words ={"able","word","excel","course","java","c#"};
      Pair<String> mm = ArrayAlg.minmax(words);
      System.out.println("first word = " + mm.getFirst());
      System.out.println("last word = " + mm.getSecond());
  • 4
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值