您的位置:首页 > 编程语言 > Java开发

Java中的泛型总结

2017-03-24 22:26 501 查看
Java中的泛型总结

定义一个坐标点类,要求能保存各种类型的数据,如:整形,浮点型,和字符串类型

既然变量类型起先不确定,那么很容易想到就是用所有类型的父类,也就是Object类来代替

不废话了,用代码来体现。


实例1:用Object来实现不确定的数据类型输入

//这是定义的坐标点类

public class Point {
private Object x;
private Object y;

// 用Object来表示不确定的类型
public Point(Object x, Object y) {
this.setX(x);
this.setY(y);
}

public void setX(Object x) {
this.x = x;
}

public Object getX() {
return x;
}

public void setY(Object y) {
this.y = y;
}

public Object getY() {
return y;
}

}

// 测试类

public class Demo {
public static void main(String[] args) {
System.out.println("用浮点数表示坐标: ");
Point p = new Point(12.23, 23.21);
// 这里把Object类转为Double类,然后自动拆箱,下面两种一样
System.out.println("X的坐标 " + (Double) p.getX());
System.out.println("Y的坐标 " + (Double) p.getY());
System.out.println();

System.out.println("用整数表示坐标: ");
Point p2 = new Point(12, 23);
System.out.println("X的坐标 " + (Integer) p2.getX());
System.out.println("Y的坐标 " + (Integer) p2.getY());
System.out.println();

System.out.println("用字符串表示坐标: ");
Point p3 = new Point("北纬29度", "东经113度");
System.out.println("X的坐标 " + (String) p3.getX());
System.out.println("Y的坐标 " + (String) p3.getY());
}

}

这样就可以代入不同类型数据了,但你别忘了,此时的数据还是Object型,也就是所有类型的父类

你必须清醒的明白自己传入的是什么类型,然后将其做向下转型处理才能使用。

结果:

用浮点数表示坐标: 

X的坐标 12.23

Y的坐标 23.21

用整数表示坐标: 

X的坐标 12

Y的坐标 23

用字符串表示坐标: 

X的坐标 北纬29度

Y的坐标 东经113度

虽然这样做满足了需求,不过却隐含了一个不安全因素,为什么说是隐含呢?

比如我们用new Point(12.23,"北纬29度")来构造一个Point对象

然后都用(Double)将其向下转型,会产生什么结果?

没错,编译会通过,但是一旦运行则会发生类型转换异常

用浮点数表示坐标: 

X的坐标 12.23

Exception in thread "main" java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Double
at com.Demo.main(Demo.java:14)

要避免类转换异常也很简单,把Object声明换成固定类型声明(如:String x,String y)即可,这样编译时就会报错

然后你就可以寻找出错的地方进行修改。不过如此一来,我们就满足不了需求了

为了达到不存在安全隐患和代入各种数据类型的目的,那些牛人们在JDK1.5当中引入了泛型这一概念

我们来看看如何用泛型改写上面的代码


实例2:泛型类

class Point<T> {

    //这里用T来表示不确定的类型

    private T x;

    private T y;

    public Point(T x, T y) {

        this.setX(x);

        this.setY(y);

    }

    public T getX() {

        return x;

    }

    public void setX(T x) {

        this.x = x;

    }

    public T getY() {

        return y;

    }

    public void setY(T y) {

        this.y = y;

    }

}

public class Demo {

    public static void main(String[] args) {

        System.out.println("用浮点数表示坐标: ");

        //用泛型改写后,使用数据无需再做向下转型处理

        Point<Double> p = new Point<Double>(12.23,23.21);

        System.out.println("X的坐标 " + p.getX());

        System.out.println("Y的坐标 " + p.getY());

        System.out.println();

 

        System.out.println("用整数表示坐标
117e3
: ");

        Point<Integer> p2 = new Point<Integer>(12, 23);

        System.out.println("X的坐标 " + p2.getX());

        System.out.println("Y的坐标 " + p2.getY());

        System.out.println();

 

        System.out.println("用字符串表示坐标: ");

        Point<String> p3 = new Point<String>("北纬29度", "东经113度");

        System.out.println("X的坐标 " + p3.getX());

        System.out.println("Y的坐标 " + p3.getY());

    }

}

使用泛型过后,可减少安全隐患的存在。

用浮点数表示坐标: 

X的坐标 12.23

Y的坐标 23.21

用整数表示坐标: 

X的坐标 12

Y的坐标 23

用字符串表示坐标: 

X的坐标 北纬29度

Y的坐标 东经113度

如果此时我们刻意传入不一样的数据类型:Point<Double> p = new Point<Double>("北纬29度",12.22);

那么,在编译时就会报错

Exception in thread "main" java.lang.Error: Unresolved compilation problem: 
The constructor Point<Double>(String, double) is undefined
at com01.Demo.main(Demo.java:7)

虽然定义了泛型,但如果你在构造函数中并未使用泛型机制的话,那么它便会把数据当作Object处理

这样做的目的主要是为了兼容JDK1.4以前的老代码,如Point p = new Point(22.11,23.21);

最终运行结果是一样的,但在编译时却会提示警告信息。


实例3:泛型方法

由上面的例子可以看到,一旦在构造方法中明确对象类型,那么整个类中就将使用同一种类型

最典型的例子是运用在集合框架里面,如:ArrayList<Integer> al = new ArrayList<Integer>();

此时,al中操作的所有对象类型便都是Integer了

可是,有时候我们并不希望固定死操作的对象,而是希望更够更加灵活的使用泛型技术

这个时候就可以尝试泛型方法
//类名后面不再定义泛型

class Print {

  //在方法中定义泛型

  public <T> void print(T t) {

      System.out.println(t);

  }

   

  public <E> void show(E e) {

      System.out.println(e);

  }

}

public class Demo {

    public static void main(String[] args) {

        Print p = new Print();

        p.print(12);

        p.print("hello");

        p.show(new Integer(33));

        p.show(23);

    }

}

其实这样一来,与在方法中使用Object对象已经没有什么太大区别了

何况,JDK1.5之后加入了自动拆装箱功能,省去了需要向下转型的麻烦
12

hello

33

23


实例4:泛型接口

//定义一个泛型接口

interface Inter<T> {
public void print(T t);

}

// 实现方式一:

class InterDemo1 implements Inter<String> {
public void print(String t) {
System.out.println("print1: " + t);
}

}

// 实现方式二:

class InterDemo2<T> implements Inter<T> {
public void print(T t) {
System.out.println("print2: " + t);
}

}

class Demo {

    public static void main(String[] args) {

        InterDemo1 id1 = new InterDemo1();

        id1.print("hello");

        InterDemo2<Integer> id2 = new InterDemo2<Integer>();

        id2.print(new Integer(23));

    }

}

实现泛型接口的方式有两种,一种是在实现的时候指定泛型类型

另一种是依然使用泛型,在构造的时候确定泛型类型

print1: hello

print2: 23
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: