目录

包装类

基本数据类型与对应的包装类

装箱和拆箱

装箱

拆箱

泛型

什么是泛型

泛型的语法与使用

泛型的编译

擦除机制

泛型的上界

泛型方法


提到泛型的话,我们就先提一下包装类吧!

包装类

在Java中,由于基本类型不是继承自Object,为了在泛型代码中可以支持基本类型,Java给每个基本类型都对应了一个包装类型。(注意int和char的包装类不是只大写第一个字母)


基本数据类型与对应的包装类

基本数据类型包装类
byteByte
shortShort
intInteger
longLong
floatFloat
doubleDouble
charCharacter
booleanBoolean

装箱和拆箱

装箱

装箱:把一个基本数据类型 转化为 包装类型的过程

装箱又分为自动装箱和显示装箱,但是都调用了valueof()方法。下面代码可以体现

eg:

public static void main(String[] args) {int a = 10;Integer b = a; //自动装箱Integer c = Integer.valueOf(a); //显示装箱}

拆箱

拆箱:把一个包装类型 转化为 基本数据类型的过程

拆箱也分为自动装箱和显示装箱。下面代码可以体现

eg:

public static void main(String[] args) {Integer a = new Integer(10);int b = a; //自动拆箱int c = a.intValue(); //显示拆箱double d = a.doubleValue();}


泛型

什么是泛型

就是适用于许多许多类型。从代码上讲,就是对类型实现了参数化


下面咱看一个问题:
实现一个类,类中包含一个数组成员,使得数组中可以存放任何类型的数据,也可以根据成员方法返回数组中某个下标的值?

class MyArray {//可以存放任何类型的话 定义一个Object数组public Object[] array = new Object[10];//pos放入的位置 val存放的数据public void set(int pos, Object val) {array[pos] = val;}//获取数据public Object get(int pos) {return array[pos];}}

但是这样写的话需要强转类型

那么怎么解决呢?

这就不得不提出泛型了


泛型的语法与使用

class 泛型类名称 {
// 这里可以使用类型参数
}


作用:

1. 泛型是将数据类型参数化,进行传递

2. 使用 表示当前类是一个泛型类。

3. 泛型目前为止的优点:数据类型参数化,编译时自动进行类型检查和转换

这样(代码体现)就能解决上面的问题了(可以不用强制类型转换)

class MyArray {//可以存放任何类型的话 定义一个Object数组public Object[] array = new Object[10];//pos放入的位置 val存放的数据public void set(int pos, T val) {array[pos] = val;}//获取数据public T get(int pos) {return (T)array[pos];}}//测试类中的main方法public static void main(String[] args) {//里面只能放包装类型MyArray myArray1 = new MyArray();MyArray myArray2 = new MyArray();}

注意:不能new一个泛型类的数组 但是可以下面这样写(不推荐)

class MyArray {public T[] array2 = new T[10]; //不能直接new(下面会解释)public T[] array3 = (T[]) new Object[10];//不建议}


泛型的编译

擦除机制

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

提出问题:
1.那为什么,T[] ts = new T[5]; 是不对的,编译的时候,替换为Object,不是相当于:Object[] ts = new Object[5]吗?

这是因为在Java中,泛型类型擦除机制会将T转换为Object,并且不支持创建泛型数组。

public class GenericArray {private T[] array;public GenericArray(int size) {this.array = new T[size]; // 编译错误}}

上述代码中,在使用new关键字创建GenericArray实例时,我们尝试创建一个大小为size的泛型数组。但是,由于类型擦除机制,编译器无法了解T的确切类型,因此无法创建泛型数组。


泛型的上界

语法:

实例:

这里的传入的类型形参(E)必须是边界(Number)的子类

求一个数组中的最大值想必大家都不陌生吧,那么大家看看下面的代码,为什么报错呢?

这里是因为呢,在编译的时候T会被替换成Object,但是在Object类中没有没有实现比较的接口,因为它是引用类型,必须 . 一个比较的方法,也得让T实现比较的接口。代码如下:

//这里extends是扩展的意思 不是继承的意思class Alg<T extends Comparable> {public T findMax(T[] array) {T max = array[0];for (int i = 1; i  0) {max = array[i];}}return max;}}

泛型方法

语法:

示例:

//泛型方法 返回值为T类型public static  T swap() {}