java包装类型详解

一、用途


在Java语言中,包装类型(Wrapper Classes)是一种特殊的类,它们将八个基本数据类型(byte、short、int、long、float、double、char、boolean)封装在一个类中。这些包装类(如Integer、Long、Double、Character等)在java.lang包中定义,提供了一系列实用的类方法和属性。尽管在很多情况下使用基本数据类型就足够了,包装类型的存在却对Java编程模式和功能的扩展至关重要。下面详细探讨Java为什么需要包装类型。

1. 对象需求

Java是一种面向对象的编程语言,这意味着操作的基本单位是对象。包装类允许程序员将基本数据类型当作对象来处理。这在编程中是非常有用的,尤其是当你需要将基本类型作为参数传递给期望对象的方法,或者需要将它们放入集合类中时。

2. 集合框架

Java的集合框架(如ArrayList、HashMap等)只能存储对象,不能存储基本数据类型。包装类解决了这一限制,使基本类型可以被存储在这些集合中。例如,要在ArrayList中存储整数,你必须使用Integer类,而不是int基本类型。

3. 提供更多方法

包装类提供了许多有用的方法来操作基本类型的数据。例如,Integer类提供了将字符串转换为整数的方法,还提供了处理整数时常用的常量和方法,如最大值、最小值等。

4. Null值的支持

基本数据类型不支持null值(它们总是有一个实际的值,比如int的默认值是0),但是在很多应用场景中,表示一个变量没有值的能力是必需的。包装类通过允许引用为null来提供这种可能性,这在数据库交互和错误处理中尤其有用。

5. 泛型支持

Java的泛型在编译时不支持基本数据类型。例如,你不能创建ArrayList<int>的实例。包装类使得可以通过使用ArrayList<Integer>来间接支持基本类型。

6. 类型安全

包装类提高了程序的类型安全,可以在编译时检测到类型错误,减少运行时的错误。这种类型检查对于构建大型、可维护的系统至关重要。

7. 自动装箱与拆箱

Java 5 引入了自动装箱与拆箱机制,使得基本类型与对应的包装类型之间可以自动转换。例如,当你将一个int赋值给一个Integer对象时,Java自动将int装箱为Integer。同样,当你将一个Integer对象赋值给一个int变量时,Java会自动拆箱。这简化了在需要对象时使用基本类型的语法。

应用案例--控制器方法

在Spring MVC中,控制器方法经常使用包装类型(如 IntegerBoolean)而不是基本类型(如 intboolean)来接收数据的原因与包装类型的特性和Web开发的需求紧密相关。下面详细探讨为什么在Spring MVC的控制器方法中更倾向于使用包装类型。

1. 允许Null值

Web应用常常需要处理来自用户输入或者外部系统的数据,这些数据可能是不完整的或者部分缺失的。基本类型如 intboolean 不能接受 null 值,它们有默认的值(如 int 的0,boolean 的false),这在某些情况下会引入潜在的错误或误解。例如,如果一个表单中的整数字段未被填写,将其映射为基本类型 int 将会得到0,这可能与用户忘记填写该字段的意图不符。使用包装类型 Integer,未填写的字段可以保持为 null,这样程序可以明确区分“未填写”和“填写了0”。

2. 类型安全

使用包装类型增加了类型安全,特别是在处理可能为 null 的数据时。这允许开发者在逻辑处理中显式检查 null,从而根据不同的情况执行不同的操作或显示不同的错误信息。这对于验证和错误处理非常有用,可以避免 NullPointerException

3. 泛型支持

Spring MVC经常使用泛型,如 List<Integer>Optional<Double> 等。泛型在Java中不支持基本数据类型。因此,为了能够使用诸如集合框架之类的泛型数据结构,控制器方法中的参数必须使用包装类型。

4. 自动装箱与拆箱

Java的自动装箱与拆箱特性使得使用包装类型和基本类型几乎无缝切换,增加了代码的灵活性。在Spring MVC中,当框架将HTTP请求参数绑定到控制器方法的参数时,这一特性尤为重要。如果HTTP请求中的某个参数缺失,对应的包装类型可以自然地被设置为 null,而不是去处理基本类型默认值可能带来的逻辑错误。

5. 灵活的数据处理

使用包装类型可以更灵活地处理数据,尤其是在需要区分缺失数据和实际数据时(如前面提到的0和 null 的区分)。这在进行数据验证、预处理、转换时特别有价值,能够让程序更加健壮。

在Spring MVC中,控制器方法使用包装类型而非基本类型的做法,主要是为了增加程序的健壮性和灵活性,以及更好地处理来自用户的输入数据。包装类型的使用允许数据保持为 null,从而提供了对特殊情况(如数据未填写、数据验证失败等)的更准确处理。此外,它还支持更广泛的编程模式,如使用集合和泛型,这在现代Web开发中极为重要。

包装类型在Java中扮演着重要的角色,不仅仅是因为它们提供了将基本数据类型用作对象的能力,还因为它们在Java的类型系统、集合框架、泛型支持、以及API中提供了增强的功能和灵活性。这些特性使Java成为一种更强大、更灵活的编程语言,能够满足现代软件开发的复杂需求。

二、原理分析


自动拆装箱

在Java中,自动装箱(autoboxing)和自动拆箱(unboxing)是Java 5引入的两个特性,它们使得基本类型(如 intdouble 等)和相应的包装类类型(如 IntegerDouble 等)之间的转换过程自动化,大大简化了程序员在处理对象时的代码编写。这两个特性背后的原理是编译器的自动转换,而不是运行时的动态转换,我们来详细了解一下这两个过程。

自动装箱(Autoboxing)

自动装箱是将基本数据类型自动转换为对应的包装类对象的过程。例如,当你将一个 int 赋值给一个 Integer 对象时,Java自动将这个基本类型转换为对象类型。

示例代码:

Integer i = 10;  // 自动装箱,将int转换为Integer

装箱过程原理:

  • 在编译阶段,编译器会识别出需要将基本类型转换为包装类对象的情形。
  • 编译器会自动插入调用包装类的 valueOf() 方法的代码,这个方法会从缓存中返回已存在的对象或创建一个新的对象。例如,对于 intInteger 的转换,编译器将上述代码转换为:
Integer i = Integer.valueOf(10);

自动拆箱(Unboxing)

自动拆箱是将包装类对象自动转换为基本数据类型的过程。例如,当你将一个 Integer 对象赋值给一个 int 变量时,Java自动将这个对象转换为基本类型。

示例代码:

Integer i = new Integer(10);
int val = i;  // 自动拆箱,将Integer转换为int

上面这两行代码对应的字节码为:

L1

    LINENUMBER 8 L1

    ALOAD 0

    BIPUSH 10

    INVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;

    PUTFIELD AutoBoxTest.i : Ljava/lang/Integer;

   L2

    LINENUMBER 9 L2

    ALOAD 0

    ALOAD 0

    GETFIELD AutoBoxTest.i : Ljava/lang/Integer;

    INVOKEVIRTUAL java/lang/Integer.intValue ()I

    PUTFIELD AutoBoxTest.n : I

    RETURN

从字节码中,我们发现装箱其实就是调用了 包装类的valueOf()方法,拆箱其实就是调用了 xxxValue()方法。

因此,

  • Integer i = 10 等价于 Integer i = Integer.valueOf(10)
  • int n = i 等价于 int n = i.intValue();

拆箱过程原理:

  • 在编译阶段,编译器会识别出需要将包装类对象转换为基本类型的情形。
  • 编译器自动插入调用对应包装类对象的 xxxValue() 方法的代码,这个方法会返回对象中存储的基本类型值。例如,对于 Integerint 的转换,编译器将上述代码转换为:

注意事项

尽管自动装箱和拆箱提供了极大的便利,它们也引入了一些需要注意的问题:

  1. 性能问题:频繁的装箱和拆箱操作可能会导致性能下降,特别是在大量计算的场景中,因为每次装箱和拆箱都可能涉及到对象的创建和方法调用。
  2. 空指针异常:在拆箱过程中,如果包装类对象为 null,拆箱时会抛出 NullPointerException。例如:
Integer i = null;
int val = i;  // 抛出NullPointerException

自动装箱和拆箱使得Java编程更加简洁,但也隐藏了一些可能影响性能和引起错误的行为。了解这些特性的背后原理可以帮助开发者更好地利用它们,同时避免可能的问题。在设计和实现涉及基本类型和包装类的系统时,应当考虑到这些因素,以确保代码的性能和健壮性。

包装类的缓存机制

Java 基本数据类型的包装类型的大部分都用到了缓存机制来提升性能。

  • Byte,Short,Integer,Long 这 4 种包装类默认创建了数值 [-128,127] 的相应类型的缓存数据
  • Character 创建了数值在 [0,127] 范围的缓存数据
  • Boolean 直接返回 True or False

Integer 缓存源码:

public static Integer valueOf(int i) {
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}
private static class IntegerCache {
    static final int low = -128;
    static final int high;
    static {
            // high value may be configured by property
            int h = 127;
            String integerCacheHighPropValue =
                sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
            if (integerCacheHighPropValue != null) {
                try {
                    int i = parseInt(integerCacheHighPropValue);
                    i = Math.max(i, 127);
                    // Maximum array size is Integer.MAX_VALUE
                    h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
                } catch( NumberFormatException nfe) {
                    // If the property cannot be parsed into an int, ignore it.
                }
            }
            high = h;

            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);

            // range [-128, 127] must be interned (JLS7 5.1.7)
            assert IntegerCache.high >= 127;
        }
}

  • 如果超出对应范围仍然会去创建新的对象,缓存的范围区间的大小只是在性能和资源之间的权衡。
  • 两种浮点数类型的包装类 Float,Double 并没有实现缓存机制。
  • 注意:所有整型包装类对象之间值的比较,尽量全部使用 equals 方法比较
  • 38
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值