基本数据类型的包装类
包装类基本知识
Java是面向对象的语言,但不是“纯面向对象”,基本数据类型就不是对象。但是我们在实际使用中经常需要将基本数据转化成对象,便于操作。 比如:Object[]数组、集合的操作中。
为了解决这个不足,在设计类时为每个基本数据类型设计了一个对应的类进行代表,这样八个和基本数据类型对应的类统称为包装类(Wrapper Class)。
包装类的作用是:实现基本数据类型、包装类对象、字符串和数字转换等。
【示例1】初识包装类
public class WrapperClassTest { public static void main(String[] args){ Integer i=new Integer(10); Integer j=new Integer(50); } } |
示例1 内存分析图
包装类均位于java.lang包,包装类和基本数据类型的对应关系如表所示:
在这八个类名中,除了Integer和Character类以外,其它六个类的类名和基本数据类型一致,只是类名的第一个字母大写即可。
在这八个类中,除了Character和Boolean都是“数字型”,数字型是java.lang.Number的子类。Number类是抽象类,因此它的抽象方法,所有子类都需要提供实现。Number类提供了抽象方法:intValue()、longValue()、floatValue()、doubleValue(),意味着所有的数字型包装类都可以互相转型。
包装类的用途
对于包装类说,这些类的用途主要包含两种:
1. 作为和基本数据类型对应的类类型存在,方便涉及到对象的操作。
2. 包含每种基本数据类型的相关属性如最大值、最小值等,以及相关的操作方法(这些操作方法的作用是在基本类型数据、包装类对象、字符串之间提供转化!)。
【示例2】包装类的使用
public class Test { /** * 测试Integer的用法,其他包装类类似 */ void testInteger(){
//基本类型转化成Integer对象 Integer int1 = new Integer(10); Integer int2 = Integer.valueOf(20); //官方推荐这种写法
//Integer对象转化成int int a = int1.intValue();
//字符串转化成Integer对象 Integer int3 = Integer.parseInt("334"); Integer int4 = new Integer("999");
//Integer对象转化成字符串 String str1 = int3.toString();
System.out.println("int能表示的最大整数:"+Integer.MAX_VALUE); //一些常见int类型相关的常量 }
} |
自动装箱和拆箱
自动装箱和拆箱就是将基本类型和包装类进行自动的互相转换。JDK5.0后,将自动装箱(autoboxing)/拆箱(unboxing)引入Java中。
自动装箱:基本类型数据处于需要对象的环境中时,会自动转为“对象”。
自动拆箱:每当需要一个值时,对象会自动转成基本类型数据, 没必要再去显式调用intValue、doubleValue()等转型方法。
自动装箱与拆箱的功能事实上是编译器来帮您的忙,编译器在编译时期依您所编写的语法,决定是否进行装箱或拆箱动作,如7-3所示。
【示例3】自动装箱
Integer i = 100; //相当于编译器自动为您作以下的语法编译: Integer i = new Integer(100); |
所以自动装箱与拆箱的功能是所谓的“编译器蜜糖”(Compiler Sugar),虽然使用这个功能很方便,但在程序运行阶段您得了解Java的语义。例如7-4所示的程序是可以通过编译的:
【示例4】包装类空指针异常问题
Integer i = null; int j = i; //这样的语法在编译时期是合法的,但是在运行时期会有错误,因为这种写法相当于: Integer i = null; int j = i.intValue(); |
null表示i没有参考至任何的对象实体,它可以合法地指定给对象参考名称。由于实际上i并没有参考至任何的对象,所以也就不可能操作intValue()方法,这样上面的写法在运行时会出现NullPointerException错误。
【示例5】自动装箱与拆箱
/** * 测试自动装箱和拆箱 * 结论:虽然很方便,但是如果不熟悉特殊情况,可能会出错! */ static void testBoxing(){ Integer b = 23; //自动装箱 int a = new Integer(20); //自动拆箱
//自动装箱和拆箱,实际上是编译器替我们完成了代码的自动编译,比如:Integer b = 23, 其实运行时执行的仍然是:Integer b = new Integer(23); //下面的问题我们需要注意: // Integer c = null; // int d = c; //此处其实就是:c.intValue(),因此抛空指针异常。 } |
·[-128,127]缓存问题
自动装箱时,对于-128-127之间的值, 为了提高效率Integer类会进行缓存处理, 这个区间的值仍然会当做基本数据类型处理,这个范围之外当做对象来处理。Integer类相关源码如下:
public static Integer valueOf(int i) { final int offset = 128; if (i >= -128 && i <= 127) { // must cache return IntegerCache.cache[i + offset]; } return new Integer(i); } |
测试代码如下:
public static void main(String[] args) { Integer h = 100; Integer u = 100; Integer h2 = 200; Integer u2 = 200;
System.out.println(h==u); //true -128-127之间,仍然会当做基本数据类型 System.out.println(h2==u2); //false 会当做真正对象处理 } |
「全栈Java笔记」是一部能帮大家从零到一成长为全栈Java工程师系列笔记。笔者江湖人称 Mr. G,10年Java研发经验,曾在神州数码、航天院某所研发中心从事软件设计及研发工作,从小白逐渐做到工程师、高级工程师、架构师。精通Java平台软件开发,精通JAVAEE,熟悉各种流行开发框架。
笔记包含从浅入深的六大部分:
A-Java入门阶段
B-数据库从入门到精通
C-手刃移动前端和Web前端
D-J2EE从了解到实战
E-Java高级框架精解
F-Linux和Hadoop