Java泛型

导语

这里是导语。对,没错,这里就是导语,没有前几次的代码情书。喜欢情书的关注我,去看我之前的Java文章吧。

主要内容

  • 泛型技术的产生背景
  • 泛型操作的实现
  • 通配符的使用
  • 泛型接口
  • 泛型方法

具体内容

泛型的引出

现在要求定义一个表示坐标的操作类(Point),在这个类里面要求保存有以下几种坐标:

  • 保存数字:x = 10、y = 20。
  • 保存小数:x = 10.2、y = 20.3。
  • 保存字符串:x = 东经20度、y = 北纬15度。

现在这个Point类设计的关键就在于x与y这两个变量的类型设计上。必须有一种类型可以保存这三类数据。首先想到的一定是Object类型:

  • int:int自动装箱为Integer,Integer向上转型为Object。
  • double:double自动装箱为Double,Double向上转型为Object。
  • String:直接向上转型为Object。

范例:初期设计如下

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

    public Objects getX() {
        return x;
    }

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

    public Objects getY() {
        return y;
    }

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

下面重复的演示三个程序,分别使用各个不同的数据类型。
在Point类里面保存整型数据

public class TestDemo {
    public static void main(String args[]) {
        // 第一步:设置数据
        Point p = new Point();
        p.setX(10);
        p.setY(20);
        // 第二步:取出数据
        int x = (Integer) p.getX();
        int y = (Integer) p.getY();
        System.out.println("x坐标:" + x + ",y坐标:" + y + "。");
    }
}

输出结果

x坐标:10,y坐标:20。

在Point类里面保存小数

public class TestDemo {
    public static void main(String args[]) {
        // 第一步:设置数据
        Point p = new Point();
        p.setX(10.2);
        p.setY(20.3);
        // 第二步:取出数据
        double x = (Double) p.getX();
        double y = (Double) p.getY();
        System.out.println("x坐标:" + x + ",y坐标:" + y + "。");
    }
}

输出结果

x坐标:10.2,y坐标:20.3。

在Point类里面保存字符串

public class TestDemo {
    public static void main(String args[]) {
        // 第一步:设置数据
        Point p = new Point();
        p.setX("东经20度");
        p.setY("北纬15度");
        // 第二步:取出数据
        String x = (String) p.getX();
        String y = (String) p.getY();
        System.out.println("x坐标:" + x + ",y坐标:" + y + "。");
    }
}

输出结果

x坐标:东经20度,y坐标:北纬15度。

此时的代码已经利用了Object数据类型解决了一切的开发问题,可是解决的关系是靠的是Object,于是失败的关键也在于Object。

范例:观察错误的代码

public class TestDemo {
    public static void main(String args[]) {
        // 第一步:设置数据
        Point p = new Point();
        p.setX("东经20度");
        p.setY(10);  // 这里改为数字
        // 第二步:取出数据
        String x = (String) p.getX();
        String y = (String) p.getY();
        System.out.println("x坐标:" + x + ",y坐标:" + y + "。");
    }
}

代码不需要执行就可以看到程序的问题,因为在存放的时候是int(Integer),而取的时候使用的是String,两个没有发生关系的对象之间要发生强制转换,就一定会产生ClassCastException。

向上转型的核心目的在于统一操作的参数上,而向下转型的目的是操作子类的特殊功能,可是现在的问题发现向下转换是一件非常不安全的操作,那么这一操作应该在代码运行之前就已经能够自动的排查出来这是最好的选择。可是之前的技术做不到。

泛型操作的实现

从JDK1.5之后开始增加了泛型技术,而泛型技术的核心意义在于:类的定义的时候,可以使用一个标记,此标记就表示类中属性或方法参数的类型标记,在使用的时候才动态的设置类型。

范例:修改代码

// 此时设置的T在Point类定义上只表示一个标记,在使用的时候需要为其设置具体的类型
public class Point<T> {  // Type = T,是一个类型,也可是任何标记
    private T x;  // 此属性的类型不知道,由Point类使用时动态决定
    private T y;  // 此属性的类型不知道,由Point类使用时动态决定
    
    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;
    }
}

在使用Point类的时候才去设置标记的内容,也就是设置了类中的属性的类型。

使用String

public class TestDemo {
    public static void main(String args[]) {
        // 第一步:设置数据
        Point<String> p = new Point<String>();
        p.setX("东经20度");
        p.setY("北纬15度");  // 如果使用p.setY(10);会报错
        // 第二步:取出数据
        String x = (String) p.getX();
        String y = (String) p.getY();
        System.out.println("x坐标:" + x + ",y坐标:" + y + "。");
    }
}

使用了泛型之后,所有类中属性的类型都是动态设置的,而所有使用泛型标记的方法参数类型也都发生改变,这样就相当于避免了向下转型的问题,从而解决了类转换的安全隐患。
但是需要特别说明的是,如果想要使用泛型,那么它能够采用的类型只能够是类,即:不能是基本类型,只能够是引用类型(可以使用Integer)。

对于泛型有两点说明:

  • 如果在使用泛型类或者是接口的时候,没有设置泛型的具体类型,那么会出现编译时的警告,同是为了保证程序不出错,所有的泛型都将使用Object表示。
  • 从JDK1.7开始可以简化声明泛型。

在JDK1.7之后实例化的时候可以写成:

Point<Integer> p = new Point<>();

通配符

为了更好的理解通配符的作用,下面先来观察一段程序。

public class Message<T> {
    private T msg;

    public T getMsg() {
        return msg;
    }

    public void setMsg(T msg) {
        this.msg = msg;
    }
}

public class TestDemo {
    public static void main(String args[]) {
        Message<String> m = new Message<String>();
        m.setMsg("Hello World !");
        fun(m);  // 引用传递
    }
    
    public static void fun(Message<String> temp) {
        System.out.println(temp.getMsg());
    }
}

输出结果

Hello World !

以上代码为Message类设置的是一个String型的泛型类型,但是如果换成其它类型就不能够使用了。

先看一组错误代码

public static void fun(Message temp) {  // 不设置接收参数为泛型,就是Object类型
    temp.setMsg("Hello");  // 如果传过来的泛型不为String就会报错
    System.out.println(temp.getMsg());
}

正确的代码如下

public static void fun(Message<?> temp) {  // 设置为通配符问号,表示不能够设置,只能够取出
    System.out.println(temp.getMsg());
}

在“?”通配符基础上还会有两个子的通配符:

  • ? extends类:设置泛型上限,可以在声明上和方法参数上使用。
    • ? extends Number:意味着可以设置Number或者Number的子类(Integer、Double...)。
  • ? super类:设置泛型下限,方法参数上使用。
    • ? super String:意味着只能够设置String或者是它的父类Object。

范例:设置泛型的上限

public class Message<T extends Number> {
    private T msg;

    public T getMsg() {
        return msg;
    }

    public void setMsg(T msg) {
        this.msg = msg;
    }
}

public class TestDemo {
    public static void main(String args[]) {
        Message<Integer> m = new Message<Integer>();
        m.setMsg(100);
        fun(m);  // 引用传递
    }
    
    public static void fun(Message<? extends Number> temp) {
        System.out.println(temp.getMsg());
    }
}

如果设置了非Number或者其子类的话,那么将出现语法错误。

范例:设置泛型下限

public class Message<T> {
    private T msg;

    public T getMsg() {
        return msg;
    }

    public void setMsg(T msg) {
        this.msg = msg;
    }
}

public class TestDemo {
    public static void main(String args[]) {
        Message<String> m = new Message<String>();
        m.setMsg("Hello");
        fun(m);  // 引用传递
    }
    
    public static void fun(Message<? super String> temp) {
        System.out.println(temp.getMsg());
    }
}

泛型接口

在之前都是将泛型定义在了一个类里面,那么泛型也可以定义在接口上声明,称为泛型接口。

范例:定义泛型接口

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

定义子类有两种形式。

形式一:在子类继续设置泛型

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

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

形式二:在子类不设置泛型,而为父接口明确的定义一个泛型类型

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

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

泛型方法

泛型方法不一定要定义在支持泛型的类里面。

范例:泛型方法定义

public class TestDemo {
    public static void main(String args[]) {
        String str = fun("Hello");
        Sysout.out.println(str.length);
    }
    
    // T的类型由传入的参数类型决定
    public static <T> T fun(T t) {
        return t;
    }
}

总结

  • 泛型解决的是向下转型所带来的安全隐患,其核心的组成就是在声明类或接口的时候不设置参数或属性的类型。
  • “?”可以接收任意的泛型类型,只能够取出,但是不能够修改。

更多内容戳这里(整理好的各种文集)

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

推荐阅读更多精彩内容

  • 我们知道,使用变量之前要定义,定义一个变量时必须要指明它的数据类型,什么样的数据类型赋给什么样的值。 假如我们现在...
    e347afbb188e阅读 295评论 0 0
  • 我们知道,使用变量之前要定义,定义一个变量时必须要指明它的数据类型,什么样的数据类型赋给什么样的值。 假如我们现在...
    今晚打肉山阅读 970评论 0 1
  • Java泛型总结# 泛型是什么## 从本质上讲,泛型就是参数化类型。泛型十分重要,使用该特性可以创建类、接口以及方...
    kylinxiang阅读 908评论 0 1
  • 开发人员在使用泛型的时候,很容易根据自己的直觉而犯一些错误。比如一个方法如果接收List作为形式参数,那么如果尝试...
    时待吾阅读 1,038评论 0 3
  • 渐渐的,你好像真的忘了自己的样子。 你忘记了坚持忘记了坚持的样子,你忘记了生活的酸甜苦辣,你忘记了你自己和身边的一...
    Black兔阅读 160评论 0 2