2022.11.29
西安
初雪…

泛型在Java中十分重要,也比较难理解,本文旨在让初学者有大概的理解。

一、什么是泛型

1.1 引入泛型

一般的类和方法,只能使用具体的类型: 要么是基本类型,要么是自定义的类。如果要编写可以应用于多种类型的代码,这种限制对代码的束缚就会很大

那我们要怎么做?

在此之前先思考这个问题:
【问题】我们能不能实现一个类,类中包含一个数组成员,使得数组中可以存放任何类型的数据,也可以根据成员方法返回数组中某个下标的值?
【思路】我们以前学过的数组,只能存放指定类型的元素,比如:

int[] array = new int[10];

那怎样让一个数组储存其他类型元素呢?
对没错,你想的是对的[doge],就是Object

众所周知,所有类的父类,默认为Object类,这样的话…把数组设置为Object不就好了?
【代码实现】

class MyArray {public Object[] array = new Object[10];public Object getArray(int pos) {return this.array[pos];}public void setArray(int pos,Object val) {this.array[pos] = val;}
}public class Test1 {public static void main(String[] args) {MyArray myArray = new MyArray();myArray.setArray(0,10);       //将整型放入数组myArray.setArray(1,"JieJie"); //将字符型放入数组String str = myArray.array[1];//errorSystem.out.println(str);}
}

运行后出现异常:

【总结】
通过上述代码我们可以得出:

  1. 任何类型数据都可以存放
  2. 数组的1下标存放的本身就是字符串,但是确编译报错。必须进行强制类型转换
String str = (String) myArray.array[1];//强制类型转换
System.out.println(str);

虽然在这种情况下,当前数组任何数据都可以存放,但是,更多情况下,我们还是希望他只能够持有一种数据类型。而不是同时持有这么多类型。所以,泛型的主要目的:就是指定当前的容器,要持有什么类型的对象(超级重要,要反复理解)。让编译器去做检查。此时,就需要把类型,作为参数传递。需要什么类型,就传入什么类型。

1.2 泛型语法

【格式】

class 泛型类名称<类型形参列表> {
// 这里可以使用类型参数
}class 泛型类名称<类型形参列表> extends 继承类/* 这里可以使用类型参数 */ {// 这里可以使用类型参数
}

【注意:】泛型只能接受类,所有的基本数据类型必须使用包装类!
【对1.1的代码用泛型改写】

class MyArray<T> {     【类名后的 <T> 代表占位符,表示当前类是一个泛型类】public T[] array = (T[]) new Object[10];     【不能直接new泛型类型的数组:意味着T[] ts = new T[5];是不对的】【可以明确告诉大家,上面这样new数组也不够完美,本文稍后会最初解释,为方便理解暂时这样new】public T getArray(int pos) {return this.array[pos];}public void setArray(int pos,T val) {this.array[pos] = val;}
}public class Test1 {public static void main(String[] args) {MyArray<Integer> myArray = new MyArray<>();    【类型后加入 <Integer> 指定当前类型】myArray.setArray(0,10);       【将整型放入数组】myArray.setArray(1,20);       【将整型放入数组】int ret = myArray.getArray(1);     【获取1位置上的数字,此时不需要进行强制类型转换】【获取元素时编译器会在编译时帮我们进行类型转换】System.out.println(ret);myArray.setArray(2,"JieJie");      【将字符型放入数组,代码编译报错,此时因为上面指定类当前的类型为Integer】【编译器会在存放元素的时候帮助我们进行类型检查】}
}

【类型推导(Type Inference)】
当编译器可以根据上下文推导出类型实参时,可以省略类型实参的填写

MyArray<Integer> myArray = new MyArray<>();

二、泛型如何编译?

上面的代码中有两句话想必也注意到了:

【获取元素时编译器会在编译时帮我们进行类型转换】
【编译器会在存放元素的时候帮助我们进行类型检查】

我们知道编译器给我们的结果是什么,可是这中间具体做了什么工作呢?
想解答这个问题,我们要先来看看擦除机制:

2.1 擦除机制

通过命令:javap -c 查看字节码文件,所有的T都是Object

在编译的过程当中,将所有的T替换为Object这种机制,我们称为:擦除机制
Java的泛型机制是在编译级别实现的。编译器生成的字节码在运行期间并不包含泛型的类型信息

也就是说运行的时候不存在所谓泛型的概念

这时候可能大家有疑问,那为什么,T[] ts = new T[5]; 是不对的,编译的时候,替换为Object,不是相当于:Object[] ts = new Object[5]吗?
下面解答:

2.2 为什么不能实例化泛型类型数组

将1.2中的getArray方法的返回值替换成为Object[]:

public Object[] getArray() { return array;
}

运行下面的主函数:

public static void main(String[] args) { MyArray<Integer> myArray1 = new MyArray<>(); Integer[] strings = myArray1.getArray();
}
//编译错误
/*Exception in thread "main" java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.Integer; at TestDemo.main(TestDemo.java:31)
*/

编译错误原因:替换后的方法将Object[]分配给Integer[]引用,程序报错

返回的Object数组里面,可能存放的是任何的数据类型,可能是String,可能是Person,运行的时
候,直接转给Integer类型的数组,发生向下转型,编译器认为是不安全的。

三、补充

public T[] array = (T[]) new Object[10];

上面的部分告诉大家,这样new数组也不够完美,在这里给出一个更好的方案:

public Object[] array2 = new Object[3];
public T getArray2 (int pos){return (T)array2[pos];
}

【Java】初识泛型(带你从初学者角度切入,通俗易懂,速进)相关推荐

  1. java object转泛型_为什么Java的泛型要用擦除实现

    在 Java 中的 泛型 ,常常被称之为 伪泛型 ,究其原因是因为在实际代码的运行中,将实际类型参数的信息擦除掉了 (Type Erasure) .那是什么原因导致了 Java 做出这种妥协的呢?下面 ...

  2. 一位久经沙场的嵌入式er站在初学者角度谈谈嵌入式开发与学习的一些问题

    一位久经沙场的嵌入式er站在初学者角度谈谈嵌入式开发与学习的一些问题 在刚刚涉足嵌入式开发的时候,总想找到这样一本书,它可以解决我一些这样那样的疑惑.但是遗憾的是,到现在也没有这样一本书面世,而且我想 ...

  3. 如何以初学者角度写好一篇国际学术论文?

    来源:AI科技评论 本文约7000字,建议阅读15分钟 本文为你分享写好一篇国际学术论文应该注意哪些方面? 人工智能顶会论文之争越来越激烈了,CVPR.AAAI.ICLR等各大会议虽然录取率逐年降低, ...

  4. 有什么好的Java自学教程视频,适合初学者

    动力节点Java培训最新上线Java实验班,等你来测试自己适不适合学习Java编程哦! 随着互联网的发展,视频教程充斥着网络,很多人为了能够在视频教程中捞取一桶金,纷纷投入视频售卖的大军之中,其中不乏 ...

  5. java使用泛型后消除泛型_如何以及何时使用泛型

    java使用泛型后消除泛型 本文是我们名为" 高级Java "的学院课程的一部分. 本课程旨在帮助您最有效地使用Java. 它讨论了高级主题,包括对象创建,并发,序列化,反射等. ...

  6. C++primer第十章 泛型算法 10.1 概述 10.2 初识泛型算法

    大多数算法都定义在头文件algorithm中.标准库还在头文件numeric中定义了 一组数值泛型算法 一般情况下,这些算法并不直接操作容器,而是遍历由两个迭代器指定的一个元素范围(参见9.2.1节, ...

  7. java 获取泛型_聊聊Java泛型擦除那些事

    >版权申明]非商业目的注明出处可自由转载 博文地址:https://blog.csdn.net/ShuSheng0007/article/details/89789849 出自:shushen ...

  8. 以初学者角度介绍TestComplete的使用

    以初学者角度介绍TestComplete的使用 2019-01-11   出处:埃森哲测试中心  作/译者:Shuyan 使用背景 在上一个Automation Project中有幸使用了TestCo ...

  9. 002 Java集合泛型面试题

    Java集合/泛型面试题 1 ArrayList和linkedList的区别 ArrayList: 可以看作是能够自动增长容量的数组 ArrayList底层的实现是Array, 数组扩容实现 Arra ...

最新文章

  1. 配置Spring数据源c3p0与dbcp
  2. USACO 1.3... 虫洞 解题报告(搜索+强大剪枝+模拟)
  3. 产品定义到产品推广的思路
  4. c++ 头文件 .h 理解与实践
  5. Git之深入解析如何使用Git的分布式工作流程与如何管理多人开发贡献的项目
  6. BigDecimal 与double 转化失真
  7. C++ 大话数据结构 09: 中缀表达式 转后缀表达式 计算器
  8. 如何在SAP Business by design的UI上扩展新的按钮
  9. bootstraptable treeGrid 懒加载_Java类加载机制及自定义加载器
  10. js图片压缩java上传,JS实现异步上传压缩图片
  11. opensource项目_最佳Opensource.com:开放的组织文化
  12. LVS(DR)+keepalived实现高可用负载均衡
  13. 和 Apple “较上劲”了?Google 解决 Cookie 隐私问题上拒绝“一刀切”
  14. stl非变易算法(一)
  15. 开源的网页防篡改监控工具推荐——WGCLOUD
  16. nas服务器搭建网站,小白都能看懂的NAS服务器搭建教程
  17. Spark History Server 架构原理介绍
  18. 移动硬盘中安装ubuntu系统——Vmware Workstation安装
  19. 微信公众号数据2019_2019年微信公众号文章数据报告
  20. 腾讯云.xb 数据库备份恢复

热门文章

  1. 使用插入排序、归并排序对链表进行排序
  2. 关于海神阁阁主真实身份的推测
  3. 一些基础的算法知识——《图解算法》
  4. 信息学奥赛一本通:1055:判断闰年
  5. 【解题报告】2021牛客寒假算法基础集训营4
  6. 构建一体化数字营销平台,助力医药企业实现销售转化能级大幅跃迁 | 案例研究
  7. 哈哈...汗汗...
  8. 社区检测(网络分析)(图聚类)---进阶概念解析
  9. yolomouse怎么用_YoloMouse(游戏鼠标光标修改工具)_YoloMouse(游戏鼠标光标修改工具)官方版下载 - 键盘鼠标 - 绿软家园...
  10. 贝尔曼-福特算法(Bellman-Ford)最短路径问题