我的编程空间,编程开发者的网络收藏夹
学习永远不晚

详细图解Java中字符串的初始化

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

详细图解Java中字符串的初始化

前言

在深入学习字符串类之前,我们先搞懂JVM是怎样处理新生字符串的。当你知道字符串的初始化细节后,再去写String s = "hello"String s = new String("hello")等代码时,就能做到心中有数。

首先得搞懂字符串常量池的概念,下面进入正文吧。

常量池

把经常用到的数据存放在某块内存中,避免频繁的数据创建与销毁,实现数据共享,提高系统性能。

八种基础数据类型除了float和double都实现了常量池技术。在近代的JDK版本中(1.7后),字符串常量池被实现在Java堆内存中。

下面通过三行代码让大家对字符串常量池建立初步认识:


public static void main(String[] args) {
    String s1 = "hello";
    String s2 = new String("hello");
    System.out.println(s1 == s2);   //false
}

先来看看第一行代码String s1 = "hello";

直接通过双引号( String s1 = "hello")声明字符串的方式,虚拟机首先会到字符串常量池中查找该字符串是否已经存在。如果存在会直接返回该引用,如果不存在则会在堆内存中创建该字符串对象,然后到字符串常量池中注册该字符串。

上面的代码中( String s1 = "hello")虚拟机首先会到字符串常量池中查找是否有存在hello字符串对应的引用。发现没有后会在堆内存创建hello字符串对象(内存地址0x0001),然后到字符串常量池中注册地址为0x0001的hello对象,也就是添加指向0x0001的引用。最后把字符串对象返回给s1。

下面看String s2 = new String("hello");

当我们使用new关键字创建字符串对象的时候,JVM将不会查询字符串常量池,它将会直接在堆内存中创建一个字符串对象,并返回给所属变量。

所以s1和s2指向的是两个完全不同的对象,判断s1 == s2的时候会返回false。

再来看下面的示例:


public static void main(String[] args) {
    String s1 = new String("hello ") + new String("world");
    s1.intern();
    String s2 = "hello world";
    System.out.println(s1 == s2);   //true
}

第一行代码String s1 = new String("hello ") + new String("world");的执行过程是这样子的:

  1. 依次在堆内存中创建hello和world两个字符串对象;
  2. 然后把它们拼接起来 (底层使用StringBuilder实现);
  3. 在拼接完成后会产生新的hello world对象,这时变量s1指向新对象hello world。

执行完第一行代码后,内存是这样子的:

第二行代码s1.intern();

当调用intern()方法时,首先会去常量池中查找是否有该字符串对应的引用,如果有就直接返回该字符串;

如果没有,就会在常量池中注册该字符串的引用,然后返回该字符串。

由于第一行代码采用的是new的方式创建字符串,所以在字符串常量池中没有保存hello world对应的引用,虚拟机会在常量池中进行注册,注册完后的内存示意图如下:

第三行代码String s2 = "hello world";

首先虚拟机会去检查字符串常量池,发现有指向hello world的引用。然后把该引用所指向的字符串直接返回给所属变量。

执行完第三行代码后,内存示意图如下:

如图所示,s1和s2指向的是相同的对象,所以当判断s1 == s2时返回true。

总结:

  • 当用new关键字创建字符串对象时,不会查询字符串常量池;
  • 当用双引号直接声明字符串对象时,虚拟机将会查询字符串常量池。

说白了就是:字符串常量池提供了字符串的复用功能,除非我们要显式创建新的字符串对象,否则对同一个字符串虚拟机只会维护一份拷贝。

反编译代码验证字符串初始化操作

下面我们再来看一个示例:


public class Main {
    public static void main(String[] args) {
        String s1 = "hello ";
        String s2 = "world";
        String s3 = s1 + s2;
        String s4 = "hello world";
        System.out.println(s3 == s4);
    }
}

首先第一行和第二行是常规的字符串对象声明,它们分别会在堆内存创建字符串对象,并会在字符串常量池中进行注册。

影响我们做出判断的是第三行代码String s3 = s1 + s2;,我们不知道s1 + s2在创建完新字符串hello world后是否会在字符串常量池进行注册。

简单点说:我们不知道这行代码是以双引号形式声明字符串,还是用new关键字创建字符串。

那么我们看下这端代码的反编译后的代码:


PS D:\code\javaSE\target\classes\demo> javap -c .\Main.class
Compiled from "Main.java"
public class demo.Main {
  public demo.Main();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: ldc           #2                  // String hello
       2: astore_1
       3: ldc           #3                  // String world
       5: astore_2
       6: new           #4                  // class java/lang/StringBuilder
       9: dup
      10: invokespecial #5                  // Method java/lang/StringBuilder."<init>":()V
      13: aload_1
      14: invokevirtual #6                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      17: aload_2
      18: invokevirtual #6                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      21: invokevirtual #7                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      24: astore_3
      25: ldc           #8                  // String hello world
      27: astore        4
      29: getstatic     #9                  // Field java/lang/System.out:Ljava/io/PrintStream;
      32: aload_3
      33: aload         4
      35: if_acmpne     42
      38: iconst_1
      39: goto          43
      42: iconst_0
      43: invokevirtual #10                 // Method java/io/PrintStream.println:(Z)V
      46: return
}

直接看重点:

  • 21: invokevirtual #7 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
  • 24: astore_3
  • 虚拟机调用StringBuilder的toString()方法获得字符串hello world,并存放至s3。

下面是我们追踪StringBuilder的toString()方法源码:


@Override
public String toString() {
    // Create a copy, don't share the array
    return new String(value, 0, count);
}

通过以上源码可以看出:s3是通过new关键字获得字符串对象的。

回到题目,也就是说字符串常量表中没有存储hello world的引用,当s4以引号的形式声明字符串时,由于在字符串常量池中查不到相应的引用,所以会在堆内存中新创建一个字符串对象。 所以s3和s4指向的不是同一个字符串对象, 结果为false。

总结

到此这篇关于Java中字符串初始化的文章就介绍到这了,更多相关Java字符串的初始化内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

详细图解Java中字符串的初始化

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

java怎么初始化字符串数组

可以使用以下方法来初始化字符串数组:1. 使用大括号({})来初始化数组,并在大括号中指定初始值:```javaString[] array = {"Hello", "World", "Java"};```2. 使用`new`关键字来创建数
2023-08-18

java字符串数组初始化的方法是什么

Java中字符串数组的初始化有多种方法,以下是其中几种常见的方式:1. 使用大括号直接初始化数组元素:javaString[] array = {"Hello", "World"};2. 使用new关键字初始化数组,并为每个元素赋值:jav
2023-10-18

Java中实例初始化和静态初始化的过程详解

Java代码初始化块是Java语言中的一个非常重要的概念。初始化块负责在创建对象时进行一些必要的操作,例如设置对象的初始状态、初始化成员变量等。初始化块被分为实例初始化块和静态初始化块两种类型。本文详细介绍了初始化的过程,需要的朋友可以参考下
2023-05-18

java 中数组初始化实例详解

1.数组初始化定义数组的方式:  int[] arr1; 或 int arr1[];数组初始化  通过上边的定义,我们只是得到了一个数组的引用。这时已经为引用分配了存储空间,但是还没有给数组对象本身分配任何空间。想要给数组对象分配存储空间
2023-05-31

Python中字符串格式化str.format的详细介绍

前言 Python 在 2.6 版本中新加了一个字符串格式化方法: str.format() 。它的基本语法是通过 {} 和 : 来代替以前的 %.。 格式化时的占位符语法:replacement_field ::= "{" [field_
2022-06-04

详解Java中类的加载与其初始化

这篇文章主要为大家详细介绍了Java中类的加载与其初始化的相关资料,文中的示例代码讲解详细,具有一定的借鉴价值,感兴趣的小伙伴可以了解一下
2022-12-15

java &amp; Android 格式化字符串详解

%1$s %1$d Android string (java & Android 格式化字符串) 1$s // String %1$d // int //R.string.old: 我今年%1$d岁了<
2022-06-06

详解bash中的初始化机制

Bash初始化文件 交互式login shell在下列情况下,我们可以获得一个login shell:uFJHrKxWJu 登录系统时获得的顶层shell,无论是通过本地终端登录,还是通过网络ssh登录。这种情况下获得的login she
2022-06-04

详解Python的字符串格式化

这篇文章主要介绍了Python的字符串格式化,python的format函数怎么用,这篇文章向大家介绍format函数用法,需要的朋友可以参考下
2023-05-16

java数组的初始化及操作详解

Java数组的初始化及操作详解:数组的初始化是指在创建数组对象时为数组元素赋予初值。Java数组的初始化有以下几种方式:1. 静态初始化:在创建数组对象时,直接为数组元素赋值。语法如下:```java数据类型[] 数组名 = {元素1, 元
2023-08-17

c语言二维字符串数组初始化的方法是什么

C语言中,可以使用以下两种方法初始化一个二维字符串数组:1. 使用字符串字面量初始化:可以使用花括号将字符串字面量括起来,并用逗号分隔每个字符串,然后将它们放入二维数组中。例如:cchar array[3][10] = { "Hello",
2023-10-20

java 字符串截取的实例详解

java 字符串截取的实例详解题目 在java中,字符串“abcd”与字符串“ab你好”的长度是一样,都是四个字符。 但对应的字节数不同,一个汉字占两个字节。 定义一个方法,按照指定的字节数来取子串。 如:对于“ab你好”,如果取三个字节,
2023-05-31

linux中shell字符串操作的详细介绍

本篇内容主要讲解“linux中shell字符串操作的详细介绍”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“linux中shell字符串操作的详细介绍”吧!在做shell批处理程序时候,经常会涉及
2023-06-09

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录