详解Java 自动装箱与拆箱的实现原理

发布时间 - 2026-01-11 00:46:48    点击率:

什么是自动装箱和拆箱

自动装箱就是Java自动将原始类型值转换成对应的对象,比如将int的变量转换成Integer对象,这个过程叫做装箱,反之将Integer对象转换成int类型值,这个过程叫做拆箱。因为这里的装箱和拆箱是自动进行的非人为转换,所以就称作为自动装箱和拆箱。原始类型byte, short, char, int, long, float, double 和 boolean 对应的封装类为Byte, Short, Character, Integer, Long, Float, Double, Boolean。

下面例子是自动装箱和拆箱带来的疑惑

 

  public class Test { 
    public static void main(String[] args) {   
      test(); 
    } 

    public static void test() { 
      int i = 40; 
      int i0 = 40; 
      Integer i1 = 40; 
      Integer i2 = 40; 
      Integer i3 = 0; 
      Integer i4 = new Integer(40); 
      Integer i5 = new Integer(40); 
      Integer i6 = new Integer(0); 
      Double d1=1.0; 
      Double d2=1.0; 

      System.out.println("i=i0\t" + (i == i0)); 
      System.out.println("i1=i2\t" + (i1 == i2)); 
      System.out.println("i1=i2+i3\t" + (i1 == i2 + i3)); 
      System.out.println("i4=i5\t" + (i4 == i5)); 
      System.out.println("i4=i5+i6\t" + (i4 == i5 + i6));   
      System.out.println("d1=d2\t" + (d1==d2));  

      System.out.println();     
    } 
  }

请看下面的输出结果跟你预期的一样吗?

输出的结果:

i=i0    true
i1=i2   true
i1=i2+i3    true
i4=i5   false
i4=i5+i6    true
d1=d2 false

为什么会这样?带着疑问继续往下看。

自动装箱和拆箱的原理

自动装箱时编译器调用valueOf将原始类型值转换成对象,同时自动拆箱时,编译器通过调用类似intValue(),doubleValue()这类的方法将对象转换成原始类型值。

明白自动装箱和拆箱的原理后,我们带着上面的疑问进行分析下Integer的自动装箱的实现源码。如下:

  public static Integer valueOf(int i) {
    //判断i是否在-128和127之间,如果不在此范围,则从IntegerCache中获取包装类的实例。否则new一个新实例
    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 final Integer cache[];

    //静态方法,类加载的时候进行初始化cache[],静态变量存放在常量池中
    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;
    }

    private IntegerCache() {}
  }

Integer i1 = 40; 自动装箱,相当于调用了Integer.valueOf(40);方法。

首先判断i值是否在-128和127之间,如果在-128和127之间则直接从IntegerCache.cache缓存中获取指定数字的包装类;不存在则new出一个新的包装类。

IntegerCache内部实现了一个Integer的静态常量数组,在类加载的时候,执行static静态块进行初始化-128到127之间的Integer对象,存放到cache数组中。cache属于常量,存放在java的方法区中。

接着看下面是java8种基本类型的自动装箱代码实现。如下:

  //boolean原生类型自动装箱成Boolean
  public static Boolean valueOf(boolean b) {
    return (b ? TRUE : FALSE);
  }

  //byte原生类型自动装箱成Byte
  public static Byte valueOf(byte b) {
    final int offset = 128;
    return ByteCache.cache[(int)b + offset];
  }

  //byte原生类型自动装箱成Byte
  public static Short valueOf(short s) {
    final int offset = 128;
    int sAsInt = s;
    if (sAsInt >= -128 && sAsInt <= 127) { // must cache
      return ShortCache.cache[sAsInt + offset];
    }
    return new Short(s);
  }

  //char原生类型自动装箱成Character
  public static Character valueOf(char c) {
    if (c <= 127) { // must cache
      return CharacterCache.cache[(int)c];
    }
    return new Character(c);
  }

  //int原生类型自动装箱成Integer
  public static Integer valueOf(int i) {
    if (i >= IntegerCache.low && i <= IntegerCache.high)
      return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
  }

  //int原生类型自动装箱成Long
  public static Long valueOf(long l) {
    final int offset = 128;
    if (l >= -128 && l <= 127) { // will cache
      return LongCache.cache[(int)l + offset];
    }
    return new Long(l);
  }

  //double原生类型自动装箱成Double
  public static Double valueOf(double d) {
    return new Double(d);
  }

  //float原生类型自动装箱成Float
  public static Float valueOf(float f) {
    return new Float(f);
  }

通过分析源码发现,只有double和float的自动装箱代码没有使用缓存,每次都是new 新的对象,其它的6种基本类型都使用了缓存策略。

使用缓存策略是因为,缓存的这些对象都是经常使用到的(如字符、-128至127之间的数字),防止每次自动装箱都创建一此对象的实例。

而double、float是浮点型的,没有特别的热的(经常使用到的)数据的,缓存效果没有其它几种类型使用效率高。

下面在看下装箱和拆箱问题解惑。

  //1、这个没解释的就是true
  System.out.println("i=i0\t" + (i == i0)); //true
  //2、int值只要在-128和127之间的自动装箱对象都从缓存中获取的,所以为true
  System.out.println("i1=i2\t" + (i1 == i2)); //true
  //3、涉及到数字的计算,就必须先拆箱成int再做加法运算,所以不管他们的值是否在-128和127之间,只要数字一样就为true
  System.out.println("i1=i2+i3\t" + (i1 == i2 + i3));//true 
  //比较的是对象内存地址,所以为false
  System.out.println("i4=i5\t" + (i4 == i5)); //false
  //5、同第3条解释,拆箱做加法运算,对比的是数字,所以为true
  System.out.println("i4=i5+i6\t" + (i4 == i5 + i6));//true   
  //double的装箱操作没有使用缓存,每次都是new Double,所以false
  System.out.println("d1=d2\t" + (d1==d2));//false

相信你看到这就应该能明白上面的程序输出的结果为什么是true,false了,只要掌握原理,类似的问题就迎刃而解了,希望对大家的学习有所帮助,也希望大家多多支持。


# java的装箱和拆箱  # java装箱和拆箱  # 原理  # java自动装箱拆箱深入剖析  # Java中的装箱和拆箱深入理解  # 浅谈Java自动装箱与拆箱及其陷阱  # Java 装箱与拆箱详解及实例代码  # 详解Java包装类及自动装箱拆箱  # java编程中自动拆箱与自动装箱详解  # Java实现拆箱和装箱的原理解析  # Java中自动装箱、拆箱引起的耗时详解  # 深入理解Java中的装箱和拆箱  # 详细说一说Java自动装箱与拆箱是什么  # 转换成  # 都是  # 的是  # 放在  # 带着  # 加载  # 他们的  # 是因为  # 浮点  # 在此  # 这就  # 我认为  # 这类  # 迎刃而解  # 不存在  # 跟你  # 有必要  # 涉及到  # 最简单  # 相信你 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: MySQL查询结果复制到新表的方法(更新、插入)  Laravel如何升级到最新版本?(升级指南和步骤)  如何用腾讯建站主机快速创建免费网站?  JavaScript如何实现继承_有哪些常用方法  韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐  JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)  长沙做网站要多少钱,长沙国安网络怎么样?  如何安全更换建站之星模板并保留数据?  Linux系统运维自动化项目教程_Ansible批量管理实战  用v-html解决Vue.js渲染中html标签不被解析的问题  网站建设整体流程解析,建站其实很容易!  Python文件异常处理策略_健壮性说明【指导】  Android仿QQ列表左滑删除操作  浅述节点的创建及常见功能的实现  如何在万网利用已有域名快速建站?  长沙企业网站制作哪家好,长沙水业集团官方网站?  如何快速生成ASP一键建站模板并优化安全性?  Laravel如何设置自定义的日志文件名_Laravel根据日期或用户ID生成动态日志【技巧】  Laravel如何处理JSON字段的查询和更新_Laravel JSON列操作与查询技巧  Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】  免费视频制作网站,更新又快又好的免费电影网站?  JS去除重复并统计数量的实现方法  INTERNET浏览器怎样恢复关闭标签页_INTERNET浏览器标签恢复快捷键与方法【指南】  中山网站制作网页,中山新生登记系统登记流程?  如何快速上传建站程序避免常见错误?  如何在云主机上快速搭建多站点网站?  ,网页ppt怎么弄成自己的ppt?  如何快速完成中国万网建站详细流程?  如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】  微信h5制作网站有哪些,免费微信H5页面制作工具?  Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】  详解vue.js组件化开发实践  Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程  在Oracle关闭情况下如何修改spfile的参数  C语言设计一个闪闪的圣诞树  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  如何用AI帮你把自己的生活经历写成一个有趣的故事?  高防服务器如何保障网站安全无虞?  php 三元运算符实例详细介绍  Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析  Laravel怎么导出Excel文件_Laravel Excel插件使用教程  Python高阶函数应用_函数作为参数说明【指导】  昵图网官网入口 昵图网素材平台官方入口  如何在香港免费服务器上快速搭建网站?  Laravel如何使用API Resources格式化JSON响应_Laravel数据资源封装与格式化输出  1688铺货到淘宝怎么操作 1688一键铺货到自己店铺详细步骤  Python制作简易注册登录系统  Java垃圾回收器的方法和原理总结