我们之前实现过的顺序表,实现的是保存某一类型的元素(如 int
型)
示例代码:
public class MyArrayList{
private int[] array; // 保存顺序表的元素,元素都为 int 类型
private int size; // 保存顺序表内存数据个数
public MyArrayList(){
this.array=new int[10];
}
public void add(int val){
// 尾插
this.array[size]=val;
this.size++;
}
public int get(int index){
// 获取 index 位置的元素
return this.array[index];
}
...
}
但是这样写的话,这个顺序表就只能存储 int
类型的元素了
如果现在需要保存指向 Person
类型对象的引用的顺序表,该如何解决呢?如果又需要保存指向 Book
类型对象的引用呢?
Object
类是 Java 中所有所有类的祖先类因此,要解决上述问题,我们可以这样做
将我们的顺序表的元素类型定义成 Object
类型,这样我们的 Object
类型的引用可以指向 Person
类型的对象或者指向 Book
类型的对象
示例代码:
public class MyArrayList{
private Object[] array; // 保存顺序表的元素,即 Object 类型的引用
private int size; // 保存顺序表内存数据个数
public MyArrayList(){
this.array=new Object[10];
}
public void add(Object val){
// 尾插
this.array[size]=val;
this.size++;
}
public Object get(int index){
// 获取 index 位置的元素
return this.array[index];
}
...
}
这样,我们就可以很自由的存储指向任意类型的对象的引用到我们的顺序表了
示例代码:
MyArrayList books = new MyArrayList();
for(int i=0; i<10;i++){
books.add(new Book()); // 插入10本书到顺序表
}
MyArrayList people = new MyArrayList();
for(int i=0; i<10; i++){
people.add(new Person()); // 插入10个人到顺序表
}
遗留问题: 现在的 MyArrayList
虽然可以做到添加任意类型的引用到其中,但会遇到下面的问题
当我们使用这样的代码时,明知道存储的是哪种类型的元素,但还是要进行强制转换。如
MyArrayList books = new MyArrayList();
books.add(1);
// 将 Object 类型转换为 int 类型 (需要类型转换才能成功)
int val=(int)books.get(0);
System.out.println(val);
// 结果为:1
虽然知道返回的元素是 int 类型,但还是要进行强制类型转换
*
创建的一个 MyArrayList
中可以存放各种类型,形成了一个大杂烩。并且将 Object 类型(具体是 A 类型)转换为 B 类型时,即使强制转换,也会产生异常 ClassCastException
MyArrayList books = new MyArrayList();
books.add(new Book());
// 将 Object 类型转换为 Person (需要类型转换才能成功)
Person person = (Person)books.get(0);
// 但是虽然编译正确了,运行时还是会抛出异常 ClassCastException
因此 Java 针对这一问题就出现了泛型
Java 泛型(generics)是 JDK 5 中引入的一个新特性,泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。
泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。
泛型可以分为两类
预备知识主要是为了学习、理解集合框架,所以这里只简单介绍泛型类,后面将会专门为泛型写一个章节。
规则:
*
在类名后面添加了类型参数声明
*
泛型类的类型参数声明部分包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符
*
泛型的泛型参数一定是类类型,如果是简单类型,那么必须是对应的包装类
这里直接将上面定义的 MyArrayList
类改写成泛型类
示例代码:
public class MyArrayList<T>{
private T[] array;
private int size;
public MyArrayList(){
this.array=(T[])new Object[10];
}
public void add(T val){
this.array[size]=val;
this.size++;
}
public T get(int index){
return this.array[index];
}
...
}
此时我们就将这个顺序表改写成了一个泛型类,接下来我们来使用它
示例代码:
MyArrayList<String> myArrayList = new MyArrayList<>();
myArrayList.add("Hello");
myArrayList.add("Goodbye");
String s = myArrayList.get(0);
System.out.println(s);
// 结果为:Hello
上述的 myArrayList
只能存放 String
类型的元素,并且不需要再添加强制类型转换
泛型的意义:
Java 中泛型标记符: 类型形参一般使用一个大写字母表示,如:
如果不重写 toString
方法,输出某个类的实例化对象,如
代码示例:
// 假设创建了一个 Person 类
Person person = new Person();
System.out.println(person);
结果为:
如果用上述的泛型类,输出其实例化对象,如
代码示例:
MyArrayList<String> myArrayList1 = new MyArrayList<>();
System.out.println(myArrayList1);
MyArrayList<Integer> myArrayList2 = new MyArrayList<>();
System.out.println(myArrayList2);
MyArrayList<Boolean> myArrayList3 = new MyArrayList<>();
System.out.println(myArrayList3);
结果为:
我们发现:
泛型类和非泛型类输出的样例格式都是一样的:类名@地址
为什么泛型类的实例化对象结果不是输出泛型类后面的泛型参数 < T >
呢?
这里就要了解泛型是怎么编译的
泛型的编译使用了一种机制:擦除机制
擦除机制只作用于编译期间,换句话说,泛型就是编译时期的一种机制,运行期间没有泛型的概念
解释:
当我们存放元素的时候,泛型就会根据 <T>
自动进行类型的检查。
*
但编译的时候,这些 <T>
就被擦除成了 Object
Object 引用可以指向任意类型的对象,但有例外出现了,8 种基本数据类型不是对象,那岂不是刚才的泛型机制要失效了?
实际上也确实如此,为了解决这个问题,Java 中引入了一类特殊的类,即这 8 种基本数据类型的包装类。在使用过程中,会将类似 int 这样的值包装到一个对象中去。
基本数据类型 | 包装类 |
---|---|
byte | Byte |
short | Short |
int | Integer |
long | Long |
float | Float |
double | Double |
char | Character |
boolean | Boolean |
Java 是一个面向对象的语言,基本类型并不具有对象的性质,为了与其他对象“接轨”就出现了包装类型
既然包装类是一个类,那么就有它对应的成员变量和成员方法。打孔大家可以具体的去查看文档了解各个包装类
包装类中有两个重要的知识点,装箱和拆箱
*
装箱: 把基本数据类型转为对应的包装类型
*
拆箱: 把包装类型转换为基本数据类型
装箱示例代码:
// 方式一
Integer i1 = 10;
// 方式二
Integer i2 = Integer.valueOf(10);
// 方式三
Integer i3 = new Integer(10);
拆箱示例代码:
// 方式一
int i = i1;
// 方式二
int i = i1.intValue();
那自动装箱又是什么呢?我们可以对下面这份代码进行反编译(反编译指令为 javap -c 类名
)
代码示例:
public class TestDemo {
public static void main(String[] args) {
Integer i = 10;
int j = i;
}
}
通过反编译指令,得到了如下结果:
Integer.valueOf
这个静态方法赋值给了 i,进行装箱操作Integer.intValue
这个方法复制给了 j,进行拆箱操作那么什么是手动装箱和手动拆箱呢?
就是和底层原理一样,通过 Integer.valueOf
和 Integer.intValue
方法进行的装箱和拆箱就是手动的
而不是通过这些方法进行的装箱和拆箱就是自动的
思考下列代码结果:
Integer a = 120;
Integer b = 120;
System.out.println(a == b);
结果为:true
再看一个代码:
Integer a = 130;
Integer b = 130;
System.out.println(a == b);
结果为:false
这是为什么呢?
首先我们看到 a 和 b 都进行了装包操作,因此我们就要去了解装包的时候发生了什么
通过转到 Integer.valueOf
的定义我们看到
该定义意思就是:如果 i 大于等于 IntegerCache
的最小值,小于它的最大值,就返回 IntegerCache.cache[i + (-IntegerCache.low)]
,否则就返回 new Integer(i)
而 new 一个对象的话,相当于比较的就是地址的值了,所以是 false
因此我们要知道 IntegerCache
的最大值以及最小值是多少,此时我们转到它的定义
上图中我们了解到 low 为 -128、high为 127,而 cache 其实就是一个数组。我们知道数组的下标是从 0 开始的,而 i + (-IntegerCache.low)
表示的最小值正好就是 0,也就是说明数组下标为 0 时存储的值就为 -128,并且依次往后递推。
因此数值在 -128 到 127 之间时返回的就是和这个数相同的值,所以结果为 true
那为什么要专门创建一个数组呢?所有数字返回 new 的对象不就行了吗?
这是因为,这样做可以提高效率。实例化对象是需要消耗资源的。而数组其实就是一个对象,可以减少资源的消耗。
版权说明 : 本文为转载文章, 版权归原作者所有 版权申明
原文链接 : https://blog.csdn.net/weixin_51367845/article/details/120818850
内容来源于网络,如有侵权,请联系作者删除!