当前位置: 首页 > news >正文

简单了解泛型和包装类

作者~小明学编程 

文章专栏Java数据结构

格言目之所及皆为回忆,心之所想皆为过往
在这里插入图片描述

目录

简单了解泛型

泛型的引入

泛型类的定义的简单演示

泛型背后作用时期和背后的简单原理

泛型类的简单使用

泛型的简单总结

包装类

基本数据类型和包装类直接的对应关系

装箱和拆箱

易错提醒


简单了解泛型

泛型的引入

我们之前实现过的顺序表,只能保存 int 类型的元素,如果现在需要保存 指向 Person 类型对象的引用的顺序表,请问应该如何解决?如果又需要保存指向 Book 对象类型的引用呢?

答:1.我们在学习多态的过程中我们知道,基类的引用是可以指向子类的对象的。

       2.我们知道Object是我们所有类的父类。

所以面对上面的问题我们就很容易想到一个解决的办法,那就是把我们的int类型改为我们的Object这样我们就能接受所有类型的数据了,接着我们就来做一个简单的实现。

class MyArrayList1 {
    public Object[] elem;
    private int userSize;
    public MyArrayList1() {
        this.elem = new Object[10];
    }
    public void add(Object x) {
        this.elem[this.userSize] = x;
        userSize++;
    }
    public Object get(int x) {
        return elem[x];
    }
}
public class TestDemo2 {
    public static void main(String[] args) {
        MyArrayList1 myArrayList = new MyArrayList1();
        myArrayList.add(10);
        myArrayList.add("ad");
        int ret = (int) myArrayList.get(0);
        System.out.println(ret);//10

    }
}

这样我们的顺序表就能接受来自不同类型的数据了。

接着我们看一下下面这段代码:

    public static void main(String[] args) {
        MyArrayList1 myArrayList = new MyArrayList1();
        myArrayList.add(10);
        myArrayList.add("ad");
        String ret = (String) myArrayList.get(0);
        System.out.println(ret);//10

    }

这里编译正确,但运行时会抛出异常 ClassCastException
提示:问题暴露的越早,影响越小。编译期间的问题只会让开发者感觉到,运行期间的错误会让所有的软件使用者承受错误风险。
所以我们需要一种机制,可以 1. 增加编译期间的类型检查 2. 取消类型转换的使用 泛型就此诞生。

泛型类的定义的简单演示

// 1. 尖括号 <> 是泛型的标志
// 2. E 是类型变量(Type Variable),变量名一般要大写
// 3. E 在定义时是形参,代表的意思是 MyArrayList 最终传入的类型,但现在还不知道
public class MyArrayList<E> {
    private E[] array;
    private int size;
    ...
}

泛型背后作用时期和背后的简单原理

1. 泛型是作用在编译期间的一种机制,即运行期间没有泛型的概念。
2. 泛型代码在运行期间,就是我们上面提到的,利用 Object 达到的效果(这里不是很准确,以后会做说明)。

泛型类的简单使用

class MyArrayList <E>{
    public E[] elem;
    private int userSize;
    public MyArrayList() {
        this.elem = (E[]) new Object[10];
    }
    public void add(E x) {
        this.elem[this.userSize] = x;
        userSize++;
    }
    public E get(int x) {
        return elem[x];
    }
}

 这里我们,看到在编译的时候就会给我们进行检查。

通过以上代码,我们可以看到泛型类的一个使用方式:只需要在所有类型后边跟尖括号,并且尖括号内是真正的类型,即 E 可以看作的最后的类型。
注意: Book 只能想象成 E 的类型,但实际上 E 的类型还是 Object。

泛型的简单总结

1. 泛型是为了解决某些容器、算法等代码的通用性而引入,并且能在编译期间做类型检查。
2. 泛型利用的是 Object 是所有类的祖先类,并且父类的引用可以指向子类对象的特定而工作。
3. 泛型是一种编译期间的机制,即 MyArrayList<Person> 和 MyArrayList<Book> 在运行期间是一个类型。
4. 泛型是 java 中的一种合法语法,标志就是尖括号 <>。

包装类

Object 引用可以指向任意类型的对象,但有例外出现了,8 种基本数据类型不是对象,那岂不是刚才的泛型机制要失效了?

实际上也确实如此,为了解决这个问题,java 引入了一类特殊的类,即这 8 种基本数据类型的包装类,在使用过程中,会将类似 int 这样的值包装到一个对象中去。

基本数据类型和包装类直接的对应关系

 除了 Integer 和 Character,剩下的就是类型的首字母大写。

装箱和拆箱

    public static void main(String[] args) {
        // 装箱操作,新建一个 Integer 类型对象,将 i 的值放入对象的某个属性中
        Integer a2 = Integer.valueOf(123);//显示装箱
        Integer a3 = new Integer(100);//显示装箱
        int b2 = a2.intValue();//显示拆箱
        // 拆箱操作,将 Integer 对象中的值取出,放到一个基本数据类型中
        Integer a = 10;//隐式 装箱
        int b = a;//隐式 拆箱
    }

可以看到在使用过程中,装箱和拆箱带来不少的代码量,所以为了减少开发者的负担,java 提供了自动机制。

    int i = 10;
    Integer ii = i; // 自动装箱
    Integer ij = (Integer)i; // 自动装箱
    int j = ii; // 自动拆箱
    int k = (int)ii; // 自动拆箱

注意:自动装箱和自动拆箱是工作在编译期间的一种机制。

易错提醒

我们看一下下面这段代码

    public static void main1(String[] args) {
        Integer a1 = 10;
        Integer a2 = 10;
        System.out.println(a1==a2);
        Integer b1 = 128;
        Integer b2 = 128;
        System.out.println(b1==b2);
    }

这段代码的输出结果是什么呢?

 没想到吧!为啥会是这样呢?

想要知道为何会是这样就得看看显示装箱的过程,这就需要我们查看源码了。

    public static Integer valueOf(int i) {
        if (i >= IntegerCache.low && i <= IntegerCache.high)
            return IntegerCache.cache[i + (-IntegerCache.low)];
        return new Integer(i);
    }

 当我们的i小于等于127并且大于等于-128的时候直接返回的是cache数组里面的值,否则的话就是要重新new()一个对象,这也就是为啥我们的b1和b2不相等的原因。

相关文章:

  • 网站设计的销售/搜索引擎优化方法
  • 做几个网站好/企业门户网站
  • wordpress做新闻网站/网络平台推广方式
  • 用dw做网站导航的步骤/浙江网站推广公司
  • wordpress编辑小工具栏/网站优化名词解释
  • 常州网站优化/域名注册商有哪些
  • 串口使用系列学习之什么是流控
  • centos7安装docker
  • 前端基石:构造函数和普通函数
  • MySQL索引-3回表查询与覆盖索引
  • 【机器学习】阿里云天池竞赛——工业蒸汽量预测(5)
  • 可编程交换机P4网络资源
  • HTTP版本 四次挥手 同源策略
  • fastapi_No.7_获取表单和文件数据
  • YOLOX 学习笔记
  • 6.链表篇2链表的介绍和实现(单/双链表)
  • myj的补作业计划HrbustOJ新生赛(struct+优先队列)
  • Day770.Redis客户端如何与服务器端交换命令和数据 -Redis 核心技术与实战