本文章参考内容如下:
- 唐大仕老师的课程
- CoreJava电子书
1. 包装类型、装箱和拆箱
"包装类型"的概念是相对于基本类型而提出的。由于Java是一门完全意义上的面向对象编程语言,所以一个经常要处理的问题是将一些基本类型的数据转为一个对象。为了达到这一效果,重写一个类的成本又太高,而Java语言恰好有这样一种类型--包装类(wrappers)来达到这样的效果。
例如,要建立一个全是整数的数组列表(ArrayList),尖括号中不能写基本类型:
List<int> ls = new ArrayList<>(); // Wrong!
根据Java语言的要求,只能写一个类在其中。所以我们面临的一个问题是将基本类型转化为一个Object
。
基本类型对应的包装类型如下表:
基本类型(primitive type) | 包装类(wrapper) |
---|---|
int | Integer |
long | Long |
float | Float |
double | Double |
short | Short |
byte | Byte |
char | Character |
boolean | Boolean |
其中,除Character
和Boolean
外,前六类都继承自Number
类。八个包装类都是final
关键字类型,意味着不能被继承!
所以,根据上面的表格来看,要建立一个全整数的ArrayList
,上面的代码应改写为:
List<Integer> ls = new ArrayList<>();
装箱(boxing):这个概念是从C#沿袭而来。上面说到,可以将包装类理解为基本类型转化为一个Object
,但这样理解并不完全准确。更为准确的是,将基本数据类型的数据作为"馅料",包进同名(全称)的包装类中。在编程语言中,基本类型和同名包装类可以很方便地转换,即"将包子皮打开,看到里面的馅料"。举例说明:
// 在ArrayList类型中,向列表中添加数据的操作是add
ls.add(3);
// 根据面向对象原理,被add的元素应该是一个Integer实例而非基本类型数据;但上面的代码中
// "3"已经被自动转化为Integer类型了,这个过程就是"装箱(boxing)"
// 上面的代码与下面的等价
ls.add(Integer.valueOf(3));
拆箱(unboxing): 与装箱正相反,是将包装类类型数据转换为基本数据类型的操作。举例说明:
// 在ArrayList类型中,获取列表中的值的方法是get
int n = ls.get(i);
在执行完get
操作之后,get
到的值是一个Integer
类型,而非int
。那为什么Integer
类型的值可以赋到int
类型的n
上呢?原因是编译器自动拆箱,将数据从Integer
这个包裹中拆解出来了。如果没有拆箱的概念,上面的代码应改写为:
int n = ls.get(i).getValue();
2. 枚举
3. 注解(From JDK 1.5)
4. "=="和"equals"方法
对于==
(双等号)而言,等号两边:
- 如果是基本类型,则判断值是否相等
- 如果是引用类型(对象类型),则判断引用是否相等
在基本类型作比较时:
- 如果是数值类型,则转换后进行比较;
- 注意:浮点数最好不要比较!因为计算机当中浮点数无论如何都有误差的
-
boolean
与int
不要直接比较 -
Double.NAN == Double.NAN
的结果是false
在枚举类型作比较时,由于内部已经进行了唯一实例化,所以可以直接判断。
在引用类型作比较时:
- 双等号比较的是引用是否相等
- 如果要判断值是否相等,则需要重写
equals
方法。 - 最好也能重写
hashCode()
方法
特别地,String
类型千万不要用==
来判断,而要用equals
方法来判断。