jvm字符串常量池_java 常量池

jvm字符串常量池_java 常量池字符串字符串字面量:就是指这个字符串本身,比如”Java”,”Hello”。字符串对象:比如newString(“abc”),或者直接Strings=”str”,后面的”str”也是一个字符串对象。字符串引用:引用就是一个变量,指向对应的字符串对象。常量池class常量池Java源文件编译之后得到的class文件,其中有项信息就是常量池,保存有字面量和符号引用,比如publicclassJ…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

字符串

字符串字面量:就是指这个字符串本身,比如”Java”,”Hello”。

字符串对象:比如new String(“abc”),或者直接String s=”str”,后面的”str”也是一个字符串对象。

字符串引用:引用就是一个变量,指向对应的字符串对象。

常量池

class常量池

Java源文件编译之后得到的class文件,其中有项信息就是常量池,保存有字面量和符号引用,比如

public class JvmClass1 {

final int b=666;

public static void main(String[] args) {

String c=”java”;

String d=”abcd”;

}

}

对应的class文件中

dc2858f39c137b17f076a7fa0df034fc.png

dd69ab841d978416a436c45827b155ae.png

这一项就是666这个字面量。

181dea0387ea4cbe5541eef35ed348db.png

55167921b697a4ad85e59e430188ecc4.png

这两项就是java和abcd这两个字符串的字面量。

而符号引用也是一些常量,比如全限定类名,字段的名称和描述符,方法的名称和描述符。

9879068a565548b385fb65df0584e44f.png

bd4a449aa864bbb06da61c5f2f6fe741.png

这是类名。

1baa1aad8d6d973768061ac09c2d01b5.png

8800622ebac305bc6218bde9e759fb6d.png

这是变量名。

常量池中有一些常量类型有index属性,指向另外一个字面量,比如CONSTANT_Class_Info,CONSTANT_String_Info等。

相同的字符串字面量在常量池中只会保存一份,例如

public class JvmClass1 {

public static void main(String[] args) {

String c=”java”;

String d=”abcd”;

String e=”java”;

String f=new String(“java”);

}

}

运行时常量池 && 字符串常量池

class常量池被加载到内存后,形成了运行时常量池,Jdk1.7之前位于方法区中,Jdk1.8之后是放在元空间,或者把元空间看做是新的方法区。

运行时常量池相对于class常量池的一个特点是具有动态性,Java不要求所有常量在编译器产生,可以在运行时产生常量加入常量池,例如String类的intern()。

String.intern

intern源码的注解是Returns a canonical representation for the string object.,意思是返回字符串对象的规范表示形式。

第二段是A pool of strings, initially empty, is maintained privately by the class,说的就是字符串常量池,JDK1.6及以前是放在方法区中,后来放到了堆中,其中保存的是字符串对象的引用,而真正的字符串对象实例是在堆中创建。

第三段是

When the intern method is invoked, if the pool already contains a string equal to this {@code String} object

as determined by the {@link #equals(Object)} method, then the string from the pool is returned. Otherwise,

this {@code String} object is added to the pool and a reference to this {@code String} object is returned.

意思是当一个字符串对象调用intern方法,如果池中已经存在值相等(通过String的equal函数比较)的字符串常量,就返回常量池中的常量,也就是堆中对应实例的引用。否则将这个字符串加入常量池。

例如

public class JvmClass1 {

public static void main(String[] args) {

String a = “hello”;

String b = new String(“hello”);

System.out.println(a == b);//false a和b是不同对象

String c = “world”;

System.out.println(c.intern() == c);//true c.intern()返回的就是”world”在常量池中的引用,和c是同一个对象

String d = new String(“mike”);

System.out.println(d.intern() == d);//false d.intern()返回的类似a,而d类似b,不同对象

String e = new String(“jo”) + new String(“hn”);

System.out.println(e.intern() == e);//true 通过拼接得到的,并没有出现”john”的字面量,所以只有当e.intern()才加入池中,所以是同一对象

String f = new String(“ja”) + new String(“va”);

System.out.println(f.intern() == f);//false 有个博客说”java”在jvm启动时自动加入字符串常量池中,不过还没找到其他什么证据。

}

}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/164771.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 我在做的测试框架

    我在做的测试框架

  • leetcode-148. 排序链表(链表排序)

    leetcode-148. 排序链表(链表排序)给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。进阶:你可以在 O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序吗?示例 1:输入:head = [4,2,1,3]输出:[1,2,3,4]示例 2:输入:head = [-1,5,3,4,0]输出:[-1,0,3,4,5]示例 3:输入:head = []输出:[] 提示:链表中节点的数目在范围 [0, 5 * 104] 内-105 <= Node.val &lt

  • android 置灰不可点击,Android Studio 运行按钮灰色的完美解决方法「建议收藏」

    android 置灰不可点击,Android Studio 运行按钮灰色的完美解决方法「建议收藏」AndroidStudio运行按钮灰色的完美解决方法今天新建项目的时候突然发现编译后运行按钮为灰色。解决方案:第一步:点击图中的AddConfiguration,出来如下界面第二步:点+号,并选择AndroidApp选项出来下图所示界面第三步:在Module中下拉框中选择app如果在Module下拉框没有app这个选项点击搜索框,输入sync,从搜索结果中选择如下项:点击运行…

  • IDEA插件-热部署:JRebel

    IDEA插件-热部署:JRebelspringboot项目开发过程中通常修改了某分部代码需要重启服务才能生效。通过JRebel插件可以实现热部署,避免了频繁重启服务。

  • 电容的基础知识

    常用电容按介质区分有纸介电容、油浸纸介电容、金属化纸介电容、云母电容、薄膜电容、陶瓷电容、电解电容等。表1 常用电容的结构和特点电容种类电 容 结 构&#

    2021年12月26日
  • leveldb多线程读写(大数据高并发解决方案)

    在并发写入的时候,leveldb巧妙的利用一个时间窗口做batch写入,这部分代码值得一读:StatusDBImpl::Write(constWriteOptions&amp;options,WriteBatch*my_batch){  //partA  Writerw(&amp;mutex_);  w.batch=my_batch;  w.sync=options.syn…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号