Java——泛型的作用

什么是向下转型和向上转型。

面向对象的转型只会发生在具有继承关系的父子类中(接口也是继承的一种)
向上转型:其核心目的在于参数的统一上,根本不需要强制类型转换。
向下转型:是为了操作子类定义的特殊功能,需要强制类型转换,可是现在存在的问题是:向下转型其实是一种非常不安全的操作,以为编译的时候,程序不会报错,而在运行的时候会报错,这就是传说中的—迷之报错。
不过呢,在JDK1.5之后,新增加了泛型的技术,这就将上述向下转型的问题消灭在了萌芽之中。
泛型的核心意义在于:类在进行定义的时候可以使用一个标记,此标记就表示类中属性或者方法以及参数的类型,标记在使用的时候,才会去动态的设置类型。
源代码如下:

package com.demo;
class Point {  //定义地置
    private Object x ;
    private Object y ; 
    public void setX(Object x) {
        this.x = x;
    }
    public void setY(Object y) {
        this.y = y;
    }
    public Object getX() {
        return x;
    }
    public Object getY() {
        return y;
    }
}
public class TestDemo {
    public static void main(String[] args) {
        //1.设置数据
        Point p = new Point() ;
        p.setX(10);
        p.setY(20);
        //2.取出数据
        int x = (Integer) p.getX();
        int y = (Integer) p.getY();
        System.out.println("x地置"+ x +",y地置"+ y);
   }
}

 

坐标代码满足字符串的:

package com.demo;
class Point {  //定义坐标
    private Object x ;
    private Object y ; 
    public void setX(Object x) {
        this.x = x;
    }
    public void setY(Object y) {
        this.y = y;
    }
    public Object getX() {
        return x;
    }
    public Object getY() {
        return y;
    }
}
public class TestDemo {
    public static void main(String[] args) {
        //1.设置数据
        Point p = new Point() ;
        p.setX("东经:100°");
        p.setY("北纬:20°");
        //2.取出数据
        String x = (String) p.getX();
        String y = (String) p.getY();
        System.out.println("x地置"+ x +",y地置"+ y);
   }
}

再来看看:泛型的基本表现形式:

package com.demo;
//此时设置的T在Point类定义上,只表示一个标记,在使用的时候需要为其设置具体的类型
class Point<T> {  //定义坐标,这个"<>"里面的东西随意,Type = T,是表示一种类型
    private T x ;  //此属性的类型不确定,由Point类使用时动态决定
    private T y ;  //此属性的类型不确定,由Point类使用时动态决定
    public void setX(T x) {
        this.x = x;
    }
    public void setY(T y) {
        this.y = y;
    }
    public T getX() {
        return x;
    }
    public T getY() {
        return y;
    }
}
//在使用Point类的时候才设置标记的内容,也就是设置了类中的属性的类型。
//下面用String试试:
public class TestDemo {
    public static void main(String[] args) {
        //1.设置数据
        Point<String> p = new Point<>() ;  //JDK1.7之后实例化的泛型可以省略。
        //加入设置的数据类型是错误的,那么在编译的时候就会自动的排查了
        p.setX("东经:10°");
        p.setY("北纬:20°");
        //2.取出数据,由于我们接收的类型就是String,所以不需要向下强制转换
        String x =  p.getX();
        String y =  p.getY();
        System.out.println("x地置"+ x +",y地置"+ y);
   }
}
//输出之后发现,所有类中属性的类型,都是动态设置的,而所有使用泛型标记的方法参数类型也都发生改变,这样
//就相当于避免了向下转型的问题,从而解决了转换的安全问题

泛型的通配符:

 

package com.demo;
class Message<T> {
    private T msg ;
    public void setMsg(T msg) {
        this.msg = msg;
    }
    public T getMsg() {
        return msg;
    }
}
public class TestDemo {
    public static void main(String[] args) {
        Message<Integer> m1 = new Message<Integer>();
        Message<String> m2 = new Message<String>();

        m1.setMsg(10);
        m2.setMsg("Hello World");
        fun(m1) ;//引用传递
        fun(m2) ;
    }
    public static void fun(Message temp){
        System.out.println(temp.getMsg());
    }
}

/*以上代码为Message类设置的是一个String型的泛型类型,但是如果说
现在设置的类型变了,那么fun()方法里面接收的"Message<String>"那么就不能够使用了,并且
fun()方法不能够针对不同的泛型进行重载,因为方法的重载认的只是参数的类型,与泛型无关
解决方法1.不设置泛型参数的泛型

在接口上必须定义其相应的子类,如果要定义子类有以下两种方式:

1.在子类上不设置泛型,但是在父类接口上要明确定义一个泛型。代码如下:

package com.demo;

interface IMessage<T>{//设置泛型接口
    public void print(T t) ;
}
//子类也继续使用泛型,并且父接口使用和子类同样的泛型标记
class MessageImpl implements IMessage<String> {

    @Override
    public void print(String t) {
        // TODO Auto-generated method stub
        System.out.println(t);
    }

}
public class TestDemo {
    public static void main(String[] args) {
        IMessage<String> msg = new MessageImpl();
        msg.print("Hello World!") ;
    }
}

2.在子类上继承实例化泛型

package com.demo;

interface IMessage<T>{//设置泛型接口
    public void print(T t) ;
}
//子类也继续使用泛型,并且父接口使用和子类同样的泛型标记
class MessageImpl<T> implements IMessage<T> {
    public void print(T t){
        System.out.println(t);
    }

}
public class TestDemo {
    public static void main(String[] args) {
        IMessage<String> msg = new MessageImpl<String>();
        msg.print("Hello World!") ;
    }
}

总结:泛型主要针对向下转型时所带来的安全隐患,其核心组成是在声明类或接口时,不设置参数或属性的类型。

 

posted @ 2020-04-17 19:08  会飞的斧头  阅读(382)  评论(0编辑  收藏  举报