zoukankan      html  css  js  c++  java
  • java中字面量,常量和变量之间的区别(附:Integer缓存机制)

    一、引子

      在各种教科书和博客中这三者经常被引用,今天复习到内存区域,想起常量池中就是存着字面量和符号引用,其实这三者并不是只在java中才有,各个语言中都有类似的定义,所以做一下总结,以示区分。

    二、例子

      先举一个简单的例子,就很容易看懂接下来的定义了。

    int a; //变量
    final int b = 10; //b为常量,10为字面量
    string str = “hello world!”; // str 为变量,hello world!为字面量

    三、各定义以及区分

    3.1 变量

      有些数据在程序运行中可以变化或者被赋值,这称为变量。例如:

    int a;
    String b;

    3.2 常量

      java中是指以final关键字修饰的变量(C语言中是constant关键字),在初始化的时候必须被赋值,并且此值和类型不能再发生改变。例如:

    final int a;
    final String b;

    3.3 字面量

      在计算机科学中,字面量(literal)是用于表达源代码中一个固定值的表示法(natation)。

      是不是还是看不懂?我一般采用一个特殊的区分方法:等号右边的八种基本类型的值、字符串值、声明为final的常量的值。例如:

    final int a = 10; //a为常量,10为字面量
    string b = “hello world!”; // b 为变量,hello world!为字面量

    四、Integer的缓存机制

    注意,有个容易误解的地方:关于Integer在-128到127之间的缓存问题。

    这个问题很多人其实都知道——如果值在-128到127之间,Integer对象的引用地址是相等的。身边很多朋友都说是字面量在常量池中缓存的原因。

    其实并不是的,这是Integer的缓存机制

    在Integer类中,定义了一个私有的静态类IntegerCache,这个类是用来支持Integer缓存的,它的作用是把一部分Integer类型的对象在Integer类加载的时候一起创建出来并放在一个cache数组中,以便以后循环使用。通常这个范围是-128-127,然而这个范围的最大值是可变的,可以通过-XX:AutoBoxCacheMax=<size>参数去修改这个值,在JVM初始化的时候,这个值被写入sun.misc.VM class系统私有配置文件中,并加载。(因为会一次性全部创建出来,所以这个数值会影响到启动性能)

    【Byte,Short,Long 有固定范围: -128 到 127。对于 Character, 范围是 0 到 127。除了 Integer 可以通过参数改变范围外,其它的都不行。】

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

    然后在调用valueOf(int i)方法的时候(自动封装就是调用这个方法),就会判断 i 是否在这个范围之内,是的话就会返回这个静态类维护的cache数组里对应的对象。所以说在这个范围内值相同的对象的引用是相同的。

    public static Integer valueOf(int i) {
      if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
      return new Integer(i);
    }

    示例:

            Integer integer1 = new Integer(121);
            Integer integer2 = 121; // 编译器编译成调用valueOf(int i)
            Integer integer3 = Integer.valueOf(121);
            
            System.out.println(integer1 == integer2);  // false
            System.out.println(integer2 == integer3);  // true

     可见new出来的对象并不会从Integer缓存中去取。

  • 相关阅读:
    阿里云Ubuntu环境搭建Docker服务
    Cocos2d-x手机游戏开发中-组合动作
    Java中将时间戳转化为Date类型
    Ubuntu14.04+eclipse下cocos2d-x3.0正式版环境的搭建
    hdu 4901 The Romantic Hero(dp)
    scikit-learn:3.4. Model persistence
    桥接模式和NAT模式差别
    JavaScript入门:004—JS凝视的写法和基本运算符
    MySQL 创建用户 与 授权
    【观点见解】解读大数据的5个误区
  • 原文地址:https://www.cnblogs.com/Xieyang-blog/p/9226077.html
Copyright © 2011-2022 走看看