《Java 核心技术卷1 第10版》学习笔记------ 对象包装器、自动装箱、拆箱

    有时, 需要将 int 这样的基本类型转换为对象。 所有的基本类型都冇一个与之对应的类。例如,Integer 类对应基本类型 int。通常, 这些类称为包装器 ( wrapper ) 这些对象包装器类拥有很明显的名字:Integer、Long、Float、Double、Short、Byte、Character 、Void 和 Boolean (前6 个类派生于公共的超类 Number)。对象包装器类是不可变的,即一旦构造了包装器,就不允许更改包装在其中的值。同时, 对象包装器类还是 final , 因此不能定义它们的子类。
    假设想定义一个整型数组列表。而尖括号中的类型参数不允许是基本类型,也就是说,不允许写成 ArrayList<int>。这里就用到了 Integer 对象包装器类。我们可以声明一个 Integer对象的数组列表。

ArrayList<Integer> list = new ArrayList<>();

警告: 由于每个值分别包装在对象中, 所以 ArrayList<lnteger> 的效率远远低于 int[ ] 数组。 因此, 应该用它构造小型集合,其原因是此时程序员操作的方便性要比执行效率更加重要。

    幸运的是, 有一个很有用的特性, 从而更加便于添加 int 类型的元素到 ArrayLisKlntegeP中。下面这个调用

list.add(3);

将自动地变换成

list.add (Integer.value0f(3));

这种变换被称为自动装箱(autoboxing。)

相反地, 当将一个 Integer 对象赋给一个 int 值时, 将会自动地拆箱。也就是说, 编译器将下列语句:

nt n = list.get(i);
// 翻译成
int n = list.get(i).intValue();

甚至在算术表达式中也能够自动地装箱和拆箱。例如,可以将自增操作符应用于一个包装器引用:

Integer n = 3;
n++;

编译器将自动地插人一条对象拆箱的指令, 然后进行自增计算, 最后再将结果装箱。

    大多数情况下,容易有一种假象, 即基本类型与它们的对象包装器是一样的,只是它们的相等性不同。大家知道, == 运算符也可以应用于对象包装器对象, 只不过检测的是对象是否指向同一个存储区域, 因此,下面的比较通常不会成立。

Integer a = 1000;
Integer b = 1000;
if (a = b) . . .

然而,Java 实现却有可能( may) 让它成立。如果将经常出现的值包装到同一个对象中,这种比较就有可能成立。这种不确定的结果并不是我们所希望的。解决这个问题的办法是在两个包装器对象比较时调用 equals 方法。

注释: 自动装箱规范要求 boolean、byte、char 127, 介于 -128 ~ 127 之间的 short 和 int 被包装到固定的对象中。例如,如果在前面的例子中将 a 和 b 初始化为 100,对它们进行比较的结果一定成立。

关于自动装箱还有几点需要说明。首先, 由于包装器类引用可以为 null, 所以自动装箱有可能会抛出一个 NullPointerException 异常:

Integer n = null;
System.out.printing * n); // Throws NullPointerException

另外, 如果在一个条件表达式中混合使用 Integer 和 Double 类型, Integer 值就会拆箱,提升为 double, 再装箱为 Double:

Integer n = 1;
Double x = 2.0;
System.out.println(true ? n : x); // Prints 1.0

最后强调一下,装箱和拆箱是编译器认可的,而不是虚拟机。编译器在生成类的字节码时, 插人必要的方法调用。虚拟机只是执行这些字节码。

使用数值对象包装器还有另外一个好处。Java 设计者发现,可以将某些基本方法放置在包装器中, 例如, 将一个数字字符串转换成数值。

要想将字符串转换成整型, 可以使用下面这条语句:

int x = Integer.parselnt(s)

这里与 Integer 对象没有任何关系, parselnt 是一个静态方法。但 Integer 类是放置这个方法的一个好地方。

警告: 有些人认为包装器类可以用来实现修改数值参数的方法, 然而这是错误的。由于 Java 方法都是值传递, 所以不可能编写一个下面这样的能够增加整型参数值的 Java 方法。

public static void triple(int x) // won't work
{
    x = 3 * x; // modifies local variable
}

将 int 替换成 Integer 又会怎样呢?

public static void triple(Integer x) // won't work
{
    x = 3 * x; // modifies local variable
}

问题是 Integer 对象是不可变的: 包含在包装器中的内容不会改变: 不能使用这些包装器类创建修改数值参数的方法。
如果想编写一个修改数值参数值的方法, 就需要使用在 org.omg.CORBA 包中定义的持有者( holder) 类型, 包括 IntHolder、BooleanHolder 等。每个持有者类型都包含一个公有 (!)域值,通过它可以访问存储在其中的值。
 

public static void triple(IntHolder x)
{
    x.value = 3 * x.value;
}

本文方法总结:

java.lang.Integer 1.0

// java.lang.Integer 1.0

int intValue( )
// 以 int 的形式返回 Integer 对象的值(在 Number 类中覆盖了 intValue方法)。

static String toString(int i )
// 以一个新 String 对象的形式返回给定数值 i 的十进制表示。

static String toString(int i ,int radix )
// 返回数值 i 的基于给定 radix 参数进制的表示。

static int parselnt(String s)
static int parseInt(String s,int radix)
// 返回字符串 s 表示的整型数值, 给定字符串表示的是十进制的整数(第一种方法,)或者是 radix 参数进制的整数(第二种方法 )。

static Integer valueOf(String s)
Static Integer valueOf(String s, int radix)
// 返回用 s 表示的整型数值进行初始化后的一个新 Integer 对象, 给定字符串表示的是十进制的整数(第一种方法,) 或者是 radix 参数进制的整数(第二种方法)。

java.text.NumberFormat 1.1

Number parse(String s)
// 返回数字值,假设给定的 String 表示了一个数值

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值