Java基础String字符串存储原理

简介: Java基础String字符串存储原理

关于java jdk中内置的一个类:java.lang.String


1、String表示字符串对象,属于引用数据类型,不属于基本数据类型

2、在java中随便使用双引号括起来的都是String对象。例如:“abc”、“def”、“HElloWorld”这3个String对象

3、java规定,双引号括起来的字符串,是不可变的,也就是说“abc”自出生至死,不可变,不能变成“abcd”,也不可能变成“ab”

4、在jdk中双引号括起来的字符串,例如“abc”、“def”都是直接存储在“方法区”的“字符串常量池”当中的

为什么SUN公司把字符串存储在一个“字符串常量池”当中呢。因为字符串在实际的开发中使用太频繁。为了

执行效率,所以把字符串放到了方法区的字符串常量池当中。


示例代码01:


public class StringTest01 {
    public static void main(String[] args) {
        //这两行代表示底层创建了3个字符串对象,都在字符串常量池当中
        String a = "abcdef";
        String b = "abcdef" + "xy";
        //分析:这是使用new的方式创建的字符串对象。这个代码中的“xy”是从哪来的?
        //凡是双引号括起来的都在字符串常量池中有一份
        //new对象的时候一定在堆内存当中开辟空间
        String c = new String("xy");
    }
}


内存分析图01:


0a2653c851af460fa595bd959398a8f1.png


示例代码02:


public class StringTest02 {
    public static void main(String[] args) {
        //"hello"是存储在方法区的字符串常量池中
        //所以这个“hello”不会创建。(因为这个对象已经存在了!)
        String s = "hello";
        String s1 = "hello";
        //分析结果是true还是false
        //双等号比较的是不是变量中保存的内存地址?是的
        System.out.println(s == s1);//true
        String s2 = new String("xy");
        String s3 = new String("xy");
        System.out.println(s2 == s3);//false
        //String类重写了equals方法,所以使用equals方法比较保险
        //通过此案例,我们知道,字符串对象之间的比较不能使用“==”
        System.out.println(s2.equals(s3));//true
        String s5 = new String("testString");
        System.out.println(s3.equals("testString"));//false存在空指针异常,不建议这样写
        //String k = null;
        //"testString"这个字符串可以后面加“.”呢?
        //因为“testString”是一个String字符串对象。只要是对象都能调用方法。
        System.out.println("testString".equals(s3));//false建议使用此方式,因为此方式可以避免空指针异常
    }
}


内存分析图02:


2d65d23f6d4748949b924e4057485923.png


示例代码03:


class User {
    private int no;
    private String name;
    public User(){
    }
    public User(int no,String name){
        this.no = no;
        this.name = name;
    }
    public int getNo() {
        return no;
    }
    public void setNo(int no) {
        this.no = no;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}
public class UserTest {
    public static void main(String[] args) {
        User u = new User(100,"张三");
    }
}


内存分析图03:


6de278e6d6694ce5bb08e7e842b7e74b.png


常见面试题:


public class StringTest03 {
    public static void main(String[] args) {
        //一共创建了3个对象
        //方法区的常量池中创建了一个对象“hello”
        //堆内存创建了两个对象
        String s1 = new String("hello");
        String s2 = new String("hello");
    }
}
相关文章
|
8月前
|
SQL JSON Java
告别字符串拼接:用Java文本块优雅处理多行字符串
告别字符串拼接:用Java文本块优雅处理多行字符串
568 108
|
10月前
|
监控 Java API
现代 Java IO 高性能实践从原理到落地的高效实现路径与实战指南
本文深入解析现代Java高性能IO实践,涵盖异步非阻塞IO、操作系统优化、大文件处理、响应式网络编程与数据库访问,结合Netty、Reactor等技术落地高并发应用,助力构建高效可扩展的IO系统。
289 0
|
10月前
|
自然语言处理 Java Apache
在Java中将String字符串转换为算术表达式并计算
具体的实现逻辑需要填写在 `Tokenizer`和 `ExpressionParser`类中,这里只提供了大概的框架。在实际实现时 `Tokenizer`应该提供分词逻辑,把输入的字符串转换成Token序列。而 `ExpressionParser`应当通过递归下降的方式依次解析
473 14
|
10月前
|
存储 缓存 安全
深入讲解 Java 并发编程核心原理与应用案例
本教程全面讲解Java并发编程,涵盖并发基础、线程安全、同步机制、并发工具类、线程池及实际应用案例,助你掌握多线程开发核心技术,提升程序性能与响应能力。
365 0
|
API C# 开发者
WPF图形绘制大师指南:GDI+与Direct2D完美融合,带你玩转高性能图形处理秘籍!
【8月更文挑战第31天】GDI+与Direct2D的结合为WPF图形绘制提供了强大的工具集。通过合理地使用这两种技术,开发者可以创造出性能优异且视觉效果丰富的WPF应用程序。在实际应用中,开发者应根据项目需求和技术背景,权衡利弊,选择最合适的技术方案。
1232 1
|
安全 Java
【Java基础面试二十七】、说一说StringBuffer和StringBuilder有什么区别
这篇文章介绍了Java中StringBuffer和StringBuilder的区别:StringBuffer是线程安全的,而StringBuilder是非线程安全的,因此在单线程环境下优先推荐使用StringBuilder以获得更好的性能。
|
安全 Java API
Java系类 之 String、StringBuffer和StringBuilder类的区别
这篇文章讨论了Java中`String`、`StringBuffer`和`StringBuilder`三个类的区别,其中`String`是不可变的,而`StringBuffer`是线程安全的可变字符串类,`StringBuilder`是非线程安全的可变字符串类,通常在单线程环境下性能更优。
Java系类 之 String、StringBuffer和StringBuilder类的区别