JAVA 常量及常量池

1.常量

常量表示程序运行过程种不可改变的值,主要作用如下:
1.代表常数,便于程序的重构和修改。
2.增加程序的可读性。
在java中,常量的语法格式只需要在变量前添加final即可。代码规范要求常量名称须用大写字母。
如:

final String USERNAME = "test";

也可以首先申明,再进行赋值,但是只能赋值一次:

final String USERNAME;
USERNAME = "test";

2.Java常量池

在java中,为了避免频繁的创建和销毁对象影响系统的性能,引入了常量池,通过常量池实现了对象的共享。
通过常量池,从而实现了以下好处:
1.节省内存空间,常量池中所有的相同对象会被合并,只占一个对象的空间。
2.节省运行时间,通过==对字符串比较,速度比equals()快。(在基本数据类型中,==比较的是数值;在复合数据类型中,比较的是内存地址。)
java的常量池可做如下分类:

2.1. 静态常量池:

即*.class文件中的常量池,class文件中的常量池不仅仅包含字符串(数字)字面量,还包含类、方法的信息,占用class文件绝大部分空间。静态常量池主要用于存放两大类常量:字面量(Literal)和符号引用量(Symbolic References),字面量相当于Java语言层面常量的概念,如文本字符串,声明为final的常量值等,符号引用则属于编译原理方面的概念,包括了如下三种类型的常量,类和接口的全限定名、字段名称和描述符、方法名称和描述符。

2.2. 运行时常量池:

是jvm虚拟机在完成类装载操作后,将class文件中的常量池载入到内存中,并保存在方法区中,我们常说的常量池,就是指方法区中的运行时常量池。
运行时常量池是方法区的一部分。在jvm1.8中,则是metaspace的一部分。
CLass文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中存放。
运行时常量池相对于CLass文件常量池的另外一个重要特征是具备动态性,Java语言并不要求常量一定只有编译期才能产生,也就是并非预置入CLass文件中常量池的内容才能进入方法区运行时常量池,运行期间也可能将新的常量放入池中,这种特性被开发人员利用比较多的就是String类的intern()方法。

3.基本数据类型包装类常量池

在java中,基本数据类型byte、char、int 、short、long、boolean的包装类 Byte、Character、Integer、Short、Long、Boolean都实现了常量池。

Integer int1 = 4;
Integer int2 = 4;
System.out.println(int1 == int2); //结果为true

正常情况下,Integer是一个对象,应该存放在堆区,但是上面的情况说明内存相等。查看Integer类源码,可以发现Integer对-128-127之间的数字做了一个缓存:

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

在静态代码区会有一个IntegerCache类,如果int数值在-128至127之间,则会写入cache。

private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer 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() {}
    }

下面通过测试验证:

    public static void main(String... args) {

        Integer int1 = 127;
        Integer int2 = 127;
        System.out.println(int1 == int2); //打印结果为true
        Integer int3 = 128;
        Integer int4 = 128;
        System.out.println(int3 == int4);//打印结果为false

    }

这样即可以发现对于前面所说的包装类也是通过其产量池进行缓存的。
需要注意的是,对于float、double的包装类Float、Double并没有实现常量池缓存技术。

    Float f1 = 1.0f;
    Float f2 = 1.0f;
    System.out.println(f1==f2); //打印结果为false

另外对于包装类型的常量池缓存问题,还有一个需要注意的地方就是对于包装类,jvm虚拟机在一些情况下会自动装箱或拆箱,如果装箱了,则内存会分配在堆区,则通过==符号返回false。如果拆箱,则会用到常量池缓存。(一般发生在赋值、方法调用等情况会产生自动装箱与拆箱。)
见如下例:

Integer i1 = 10;
Integer i2 = 10;
Integer i3 = 0;
Integer i4 = new Integer(10);
Integer i5 = new Integer(10);
Integer i6 = new Integer(0);
System.out.println(i1==i2); //true 
System.out.println(i1==i2+i3);//true
System.out.println(i1==i4); //false
System.out.println(i4==i5); //false
System.out.println(i4==i5+i6);//true 运算导致等号右边是int 因此会将等号左边拆箱进行比较
System.out.println(10==i5+i6);//true

4.java字符串常量池

在java中,用String表示字符串。查看String的的源码,可以发现,Stirng都是由char数组构成的产量。

 private final char value[];

jvm为了避免重复创建过多的String对象,因此也会用常量池就行优化。在java代码中,所有代码中的字符串会被加入常量池。所有new的对象,会重新在堆内存中分配空间。String可以通过如下两种方式进行创建。

  String str1 = "abcd";//直接在常量池中得到对象
  String str2 = new String("abcd");//在堆中创建对象
  System.out.println(str1==str2);//false

需要注意如下几点:
1.jvm在编译的过程中,会对java代码进行优化,对于String字符串,凡是使用+号而又没有其他变量参与的表达式,都会合并为一个字符串并写入常量池。
2.对于所有包含new方式新建的对象(保含null)的+号连接的表达式,所产生的新对象不会放入常量池。
2.intern 方法可以将堆中的对象设置到常量池。
参考如下demo

String s1 = "hello";
String s2 = "hel" + "lo";
String s3 = "hel";
String s4 = s3+"lo";
String s5 = new String("hello");
String s6 = "hello"+null;
String s7 = new String("hel");
String s8 = s7 + "lo";
System.out.println(s1 == s2); // true 带+表达是被jvm优化合并为一个,常量池比较
System.out.println(s1 == s4);//false 表达式有变量,无法优化,新值在堆中
System.out.println(s1 == s5);//false s5是堆中 s1在常量池
System.out.println(s1 == s6);//false 表达式有null无法优化 其结果在堆中重新分配
System.out.println(s1 == s8);//false 堆中重新分配
String s9 = new String("hello").intern();
System.out.println(s1 == s9);//通过intern重新写入常量池
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,497评论 18 399
  •   需要说明的一点是,这篇文章是以《深入理解Java虚拟机》第二版这本书为基础的,这里假设大家已经了解了JVM的运...
    Geeks_Liu阅读 13,966评论 5 44
  • 转自:http://blog.csdn.net/jackfrued/article/details/4492194...
    王帅199207阅读 8,474评论 3 93
  • 人心里住着两个人吧 一个对自己说:“算了,今日不要写文了吧。” 另一个反驳道:“怎么可以呢,你怎么又变的懒惰...
    若鱼爱画画阅读 212评论 0 1
  • 我是一个某品牌的奢侈品导购,长得不算精致动人,也算是对得起客人,口才不能说伶牙俐齿,但让你买我手上的货也是轻...
    醉蚊子阅读 142评论 0 1