Java集合框架入门之泛型和包装类

目录
  • 1. 预备知识-泛型(Generic)
    • 1.1 泛型的引入
    • 1.2 泛型的分类
    • 1.3 泛型类的定义
    • 1.4 泛型编译的机制
  • 2. 预备知识-包装类(Wrapper Class)
    • 2.1 基本数据类型和包装类的对应关系
    • 2.2 包装类介绍
    • 2.3 装箱(boxing)和拆箱(unboxing)
    • 2.4 自动装箱(autoboxing)和自动拆箱(autounboxing)
    • 2.5 包装类面试题

前言: 本章主要是为了后面学习集合框架所做的知识补充。补充了泛型以及包装类两个知识,但是该章泛型的讲解不够全面,主要是为了集合框架学习做铺垫。

1. 预备知识-泛型(Generic)

1.1 泛型的引入

我们之前实现过的顺序表,实现的是保存某一类型的元素(如 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 中引入的一个新特性,泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。

泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。

1.2 泛型的分类

泛型可以分为两类

  • 泛型类
  • 泛型方法

预备知识主要是为了学习、理解集合框架,所以这里只简单介绍泛型类,后面将会专门为泛型写一个章节。

1.3 泛型类的定义

规则:

  • 在类名后面添加了类型参数声明
  • 泛型类的类型参数声明部分包含一个或多个类型参数,参数间用逗号隔开。一个泛型参数,也被称为一个类型变量,是用于指定一个泛型类型名称的标识符
  • 泛型的泛型参数一定是类类型,如果是简单类型,那么必须是对应的包装类

这里直接将上面定义的 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 中泛型标记符: 类型形参一般使用一个大写字母表示,如:

  • E — Element(在集合中使用,因为集合中存放的是元素)
  • T — Type(Java 类)
  • K — Key(键)
  • V — Value(值)
  • N — Number(数值类型)
  • ? —表示不确定的 Java 类型

1.4 泛型编译的机制

如果不重写 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

2. 预备知识-包装类(Wrapper Class)

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

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

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

基本数据类型 包装类
byte Byte
short Short
int Integer
long Long
float Float
double Double
char Character
boolean Boolean

2.2 包装类介绍

Java 是一个面向对象的语言,基本类型并不具有对象的性质,为了与其他对象“接轨”就出现了包装类型

既然包装类是一个类,那么就有它对应的成员变量和成员方法。打孔大家可以具体的去查看文档了解各个包装类

2.3 装箱(boxing)和拆箱(unboxing)

包装类中有两个重要的知识点,装箱和拆箱

  • 装箱: 把基本数据类型转为对应的包装类型
  • 拆箱: 把包装类型转换为基本数据类型

装箱示例代码:

// 方式一
Integer i1 = 10;
// 方式二
Integer i2 = Integer.valueOf(10);
// 方式三
Integer i3 = new Integer(10);

拆箱示例代码:

// 方式一
int i = i1;
// 方式二
int i = i1.intValue();

2.4 自动装箱(autoboxing)和自动拆箱(autounboxing)

那自动装箱又是什么呢?我们可以对下面这份代码进行反编译(反编译指令为 javap -c 类名

代码示例:

public class TestDemo {
    public static void main(String[] args) {
        Integer i = 10;
        int j = i;
    }
}

通过反编译指令,得到了如下结果:

  • 我们发现在底层中 10 是通过 Integer.valueOf 这个静态方法赋值给了 i,进行装箱操作
  • 再将 i 通过 Integer.intValue 这个方法复制给了 j,进行拆箱操作

那么什么是手动装箱和手动拆箱呢?

就是和底层原理一样,通过 Integer.valueOfInteger.intValue 方法进行的装箱和拆箱就是手动的

而不是通过这些方法进行的装箱和拆箱就是自动的

2.5 包装类面试题

思考下列代码结果:

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 的对象不就行了吗?

这是因为,这样做可以提高效率。实例化对象是需要消耗资源的。而数组其实就是一个对象,可以减少资源的消耗。

到此这篇关于Java集合框架入门之泛型和包装类的文章就介绍到这了,更多相关Java 泛型详解内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java基础开发泛型类的详解

    目录 前言 泛型概念 泛型类 结论 前言 在软件开发中,有许多执行过程很类似,许多人使用复制粘贴完成功能,这种做法虽然编译器不会报错,但会使用波浪线给出提示,给以后的维护带来了很大的隐患.这种情况开发人员通常根据需要成员抽取公用方法.公用类或使用继承完成,提高了代码的复用.但是,在一些特殊情况(如执行过程中会使用到对象,这些对象操作相同,但具体的模块有有所区别),此时只能使用泛型完成代码的复用. 泛型概念 所谓泛型就是将类型由原来的具体类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式

  • Java基础之匿名内部类、包装类

    目录 1.匿名内部类 2.Object类简介 2.1 取得对象信息toString() 2.2 对象的比较equals() 2.3 Object接口引用数据类型 3.包装类 3.1 装箱与拆箱 3.2 字符串与基本数据类型的转换 3.3 包的定义 3.4 包的导入 4.访问控制权限 5.jar命令 1.匿名内部类 内部类:在一个类的内部定义了另外的类,称为内部类,匿名内部类指的是没有名字的内部类.为了清楚内部类的主要作用,下面首先观察一个代码. interface IMessage{ publi

  • JAVA基本类型包装类 BigDecimal BigInteger 的使用

    目录 1.了解包装类 2.Integer 3.Double 4.BigDecimal 5.BigInteger 1.了解包装类 Java 中预定义了八种基本数据类型,包括:byte,int,long,double,float,boolean,char,short.基本类型与对象类型最大的不同点在于,基本类型基于数值,对象类型基于引用. 例如有一个方法 f() ,它的参数分别是对象类型 和 基本类型: void f(Object obj){ //参数引用类型,保存的是内存地址 } f(123){

  • Java语法关于泛型与类型擦除的分析

    泛型与类型擦除 泛型,JDK 1.5新特性,本质是参数化类型(Parametersized Type) 的应用,即所操作的数据类型被指定为一个参数.这种参数类型可用在: 类 接口 方法 的创建中, 分别称为: 泛型类 泛型接口 泛型方法 在Java还没有泛型的版本时.只能通过: Object 是所有类型的父类 类型强制转换 两个特性协作实现类型泛化.例如,在哈希表的存取中,JDK 1.5之前使用HashMap的get() 方法,返回值就是个Object.由于Java语言里面所有的类型都维承于ja

  • Java中的纸老虎之泛型

    目录 一. 泛型的定义 二. 为什么要用到泛型 三. 泛型的写法 四. 泛型的使用实例 1. 求最大值 2. 优化 五. 通配符 1. 基本写法 2. 上界 3. 下界 六. 泛型的限制 泛型,其实算是Java当中比较难的语法了,很多人一开始都对其一知半解,也很害怕阅读带泛型的源码,虽然看起来语法很难,但当你理解后会觉得很简单,其实只是一个纸老虎罢了.下面,我将会用非常简单易懂的方式带你去理解它,相信你在认真看完后会有非常大的收获,从此不会再畏惧它! 一. 泛型的定义 这里大家可以不必去看网上的

  • Java包装类之自动装箱与拆箱

    JDK 5.0之前 基本数据类型<---->包装类:调用包装类的构造器(代码里有知识点和注意点) 转换目的:有了类的特点,就可以调用类中的方法 public class WrapperTest { // 基本数据类型--->包装类:调用包装类的构造器 @Test public void test1() { int num1 = 10; Integer in1 = new Integer(num1); System.out.println(in1.toString());//10 Inte

  • Java集合框架入门之泛型和包装类

    目录 1. 预备知识-泛型(Generic) 1.1 泛型的引入 1.2 泛型的分类 1.3 泛型类的定义 1.4 泛型编译的机制 2. 预备知识-包装类(Wrapper Class) 2.1 基本数据类型和包装类的对应关系 2.2 包装类介绍 2.3 装箱(boxing)和拆箱(unboxing) 2.4 自动装箱(autoboxing)和自动拆箱(autounboxing) 2.5 包装类面试题 前言: 本章主要是为了后面学习集合框架所做的知识补充.补充了泛型以及包装类两个知识,但是该章泛型

  • 新手初学Java集合框架

    目录 Java集合框架 集合 List接口 ArrayList Vector LinkedList: 泛型: Set接口 HashSet TreeSet Map接口 特点: 遍历: HashMap Hashtable TreeMap 总结 Java集合框架 集合 概念:对象的容器,定义了对多个对象进行操作的常用方法.可实现数组的功能. 集合和数组的区别: 数组长度固定,集合长度不固定 数组可以存储基本类型和引用类型,集合只能存储引用类型. 测试 /* 1.添加 2.删除 3.遍历 4.判断 */

  • 关于Java集合框架面试题(含答案)上

    1.Java集合框架是什么?说出一些集合框架的优点? 每种编程语言中都有集合,最初的Java版本包含几种集合类:Vector.Stack.HashTable和Array.随着集合的广泛使用,Java1.2提出了囊括所有集合接口.实现和算法的集合框架.在保证线程安全的情况下使用泛型和并发集合类,Java已经经历了很久.它还包括在Java并发包中,阻塞接口以及它们的实现.集合框架的部分优点如下: (1)使用核心集合类降低开发成本,而非实现我们自己的集合类. (2)随着使用经过严格测试的集合框架类,代

  • 关于Java集合框架的总结

    本篇文章先从整体介绍了Java集合框架包含的接口和类,然后总结了集合框架中的一些基本知识和关键点,并结合实例进行简单分析.当我们把一个对象放入集合中后,系统会把所有集合元素都当成Object类的实例进行处理.从JDK1.5以后,这种状态得到了改进:可以使用泛型来限制集合里元素的类型,并让集合记住所有集合元素的类型. 一.综述 所有集合类都位于java.util包下.集合中只能保存对象(保存对象的引用变量).(数组既可以保存基本类型的数据也可以保存对象). 当我们把一个对象放入集合中后,系统会把所

  • 关于Java集合框架面试题(含答案)下

    21.HashMap和HashTable有何不同? (1)HashMap允许key和value为null,而HashTable不允许. (2)HashTable是同步的,而HashMap不是.所以HashMap适合单线程环境,HashTable适合多线程环境. (3)在Java1.4中引入了LinkedHashMap,HashMap的一个子类,假如你想要遍历顺序,你很容易从HashMap转向LinkedHashMap,但是HashTable不是这样的,它的顺序是不可预知的. (4)HashMap

  • JAVA集合框架专题

    一.Java集合框架概述 集合可以看作是一种容器,用来存储对象信息.所有集合类都位于java.util包下,但支持多线程的集合类位于java.util.concurrent包下. 数组与集合的区别如下: (1)数组长度不可变化而且无法保存具有映射关系的数据:集合类用于保存数量不确定的数据,以及保存具有映射关系的数据. (2)数组元素既可以是基本类型的值,也可以是对象:集合只能保存对象. Java集合类主要由两个根接口Collection和Map派生出来的,Collection派生出了三个子接口:

  • Java集合框架超详细小结

    目录 一:Collection集合 1.1集合概述: 1.2集合架构 1.3Collection集合常用方法 二:迭代器Iterator 2.1Iterator接口 2.2Iterator的实现原理: 2.3增强for() 2.4迭代器注意事项 三:泛型 3.1泛型概述 3.2泛型的优缺点 3.3泛型的定义与使用 泛型方法 泛型接口 3.4泛型的通配符 通配符高级使用-----受限泛型 四:Java常见数据结构 4.1栈 4.2队列 4.3数组 4.4链表 4.5红黑树 五:List集合体系 5

  • Java集合框架中迭代器Iterator解析

    Java里面的数组数据可以通过索引来获取,那么对象呢?也是通过索引吗?今天我们就来分析一下Java集合中获取集合对象的方法迭代-Iterator. 本篇文章主要分析一下Java集合框架中的迭代器部分,Iterator,该源码分析基于JDK1.8,分析工具,AndroidStudio,文章分析不足之处,还请指正! 一.简介 我们常常使用 JDK 提供的迭代接口进行 Java 集合的迭代. Iterator iterator = list.iterator(); while(iterator.has

  • Java集合框架LinkedList详解及实例

    Java集合框架LinkedList详解 LinkedList定义 package java.util; public class LinkedList<E> extends AbstractSequentialList<E> implements List<E>, Deque<E>, Cloneable, java.io.Serializable{ transient int size = 0; transient Node<E> first;

  • 详谈java集合框架

    1.为什么使用集合框架 当我们并不知道程序运行时会需要多少对象,或者需要更复杂方式存储对象--可以使用Java集合框架 2.Java集合框架包含的内容 接口:(父类)Collection接口下包含List(子类 )接口和Set(子类) 接口 List接口下又包含(ArrayList集合实现类和LinkedList集合实现类) Set接口下又包含(HashSet集合实现类和TreeSet集合实现类) 接口:(父类)Map接口下包含(HashMap集合实现类和TreeMap 集合实现类) *Coll

随机推荐