一些Java高级语法
本文章参考内容如下:
- 唐大仕老师的课程
- 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方法来判断。