String s = new String("java"); 只创建了一个对象!好好看好好学,要了解原理!

简介: String s = new String("java"); 只创建了一个对象!好好看好好学,要了解原理!
public static void main(String[] args) {
    String str1 = new StringBuilder("计算机").append("软件").toString();
    System.out.println(str1.intern() == str1);

    String str2 = new StringBuilder("ja").append("va").toString();
    System.out.println(str2.intern() == str2);
}

这个题目是在《深入理解Java虚拟机:JVM高级特性与实战(第3版)》看到的,按理说这两个结果不应该一样吗?为什么会出这种题呢?

640.jpg
640-1.jpg

作者想靠诉我,一次不信,两次你应该信了吧!机智如我。为了证明我如此机智,我特地的试了一下。

640.png

额,应该我看错了,再来亿遍。。。我的idea肯定出问题了,让同事跑一下试试!

640.jpg

为什么?为什么?为什么?你要离开我!(不好意思串台了)。为什么是true和flase。

640.jpg

吧啦吧啦吧啦说了一大堆,意思就是想让读者知道,字符串常量池jdk1.6和jdk1.7及以上存放的位置变啦,以前放到永久代的,以后就放在java堆上了。jdk1.6的intern()的方法首次遇到会把字符串复制一份到有永久代,而jdk1.7及以上的只需要在常量池里记录一下首次出现的实例引用。"只需要在常量池里记录一下首次出现的实例引用"说白了就是常量池记录第一次出现这个字符串的地址。其实我这地方有个好奇,那不就是这个对象永久不会被垃圾收集器收集了嘛?当然我也不敢问。。。

既然都说到这了,那就一起看看intern()方法吧!(jdk1.8)

640.jpg

/**
    * Returns a canonical representation for the string object.
    * <p>
    * A pool of strings, initially empty, is maintained privately by the
    * class {@code String}.
    * <p>
    * 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.
    * <p>
    * It follows that for any two strings {@code s} and {@code t},
    * {@code s.intern() == t.intern()} is {@code true}
    * if and only if {@code s.equals(t)} is {@code true}.
    * <p>
    * All literal strings and string-valued constant expressions are
    * interned. String literals are defined in section 3.10.5 of the
    * <cite>The Java&trade; Language Specification</cite>.
    *
    * @return  a string that has the same contents as this string, but is
    *          guaranteed to be from a pool of unique strings.
    */
public native String intern();

原谅我看不懂,所以谷歌翻译了下。其实意思就是调用intern方法时,如果池中已经包含一个等于此对象的字符串,此方法,则从池中取出字符串返回。否则,此对象地址将添加到池,并返回对此对象的引用。 额,和上面一个意思。

叽叽歪歪说了半天,我知道第一个是true,我是问你第二个为什么是false,光说屁话。

640.jpg

老爷们别生气,我这就慢慢给你道来。其实我也不知道。然后在知乎看到一片文章《如何理解《深入理解java虚拟机》第二版中对String.intern()方法的讲解中所举的例子?》,呵呵哒,原来在初始化的时候已经把java这个字符串放到常量池中了。

640.jpg

你说这不巧了吗?

640.jpg

举个例子让我怀疑了人生,哎。。。
哎,别先,如果我发现了一个面试题,String s = new String("s")证明他是创建两个对象的了,哈哈哈哈。。。

640-1.jpg

public static void main(String[] args) {
    String s = new String("s");
    System.out.println(s.intern() == s);
}

结果是false,那不就是创建了两个对象吗?第一个是new String("s"),第二个是常量池的s。哈哈哈哈。
小编小编,那你怎么证明java启动的时候常量池里面没有s呢?毕竟java这个字符串一开始也存在常量池啊!
额。。。

640.jpg

我又想到一个有意思的问题,String s = new String("java")创建了几个对象,哈哈哈哈,它创建了一个对象,这次没人反驳我了把。哈哈哈哈~

相关文章
|
1月前
|
存储 Java 关系型数据库
高效连接之道:Java连接池原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。频繁创建和关闭连接会消耗大量资源,导致性能瓶颈。为此,Java连接池技术通过复用连接,实现高效、稳定的数据库连接管理。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接池的基本操作、配置和使用方法,以及在电商应用中的具体应用示例。
64 5
|
5天前
|
存储 JavaScript Java
Java 中的 String Pool 简介
本文介绍了 Java 中 String 对象及其存储机制 String Pool 的基本概念,包括字符串引用、构造方法中的内存分配、字符串文字与对象的区别、手工引用、垃圾清理、性能优化,以及 Java 9 中的压缩字符串特性。文章详细解析了 String 对象的初始化、内存使用及优化方法,帮助开发者更好地理解和使用 Java 中的字符串。
Java 中的 String Pool 简介
|
11天前
|
缓存 安全 Java
java 为什么 String 在 java 中是不可变的?
本文探讨了Java中String为何设计为不可变类型,从字符串池的高效利用、哈希码缓存、支持其他对象的安全使用、增强安全性以及线程安全等方面阐述了不可变性的优势。文中还通过具体代码示例解释了这些优点的实际应用。
java 为什么 String 在 java 中是不可变的?
|
22天前
|
JSON Java 关系型数据库
Java更新数据库报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
在Java中,使用mybatis-plus更新实体类对象到mysql,其中一个字段对应数据库中json数据类型,更新时报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
35 4
Java更新数据库报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
|
5天前
|
存储 Java
Java 11 的String是如何优化存储的?
本文介绍了Java中字符串存储优化的原理和实现。通过判断字符串是否全为拉丁字符,使用`byte`代替`char`存储,以节省空间。具体实现涉及`compress`和`toBytes`方法,前者用于尝试压缩字符串,后者则按常规方式存储。代码示例展示了如何根据配置决定使用哪种存储方式。
|
1月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
1月前
|
Java
Java之CountDownLatch原理浅析
本文介绍了Java并发工具类`CountDownLatch`的使用方法、原理及其与`Thread.join()`的区别。`CountDownLatch`通过构造函数接收一个整数参数作为计数器,调用`countDown`方法减少计数,`await`方法会阻塞当前线程,直到计数为零。文章还详细解析了其内部机制,包括初始化、`countDown`和`await`方法的工作原理,并给出了一个游戏加载场景的示例代码。
Java之CountDownLatch原理浅析
|
1月前
|
Java 索引 容器
Java ArrayList扩容的原理
Java 的 `ArrayList` 是基于数组实现的动态集合。初始时,`ArrayList` 底层创建一个空数组 `elementData`,并设置 `size` 为 0。当首次添加元素时,会调用 `grow` 方法将数组扩容至默认容量 10。之后每次添加元素时,如果当前数组已满,则会再次调用 `grow` 方法进行扩容。扩容规则为:首次扩容至 10,后续扩容至原数组长度的 1.5 倍或根据实际需求扩容。例如,当需要一次性添加 100 个元素时,会直接扩容至 110 而不是 15。
Java ArrayList扩容的原理
|
20天前
|
Java
在Java中如何将基本数据类型转换为String
在Java中,可使用多种方法将基本数据类型(如int、char等)转换为String:1. 使用String.valueOf()方法;2. 利用+运算符与空字符串连接;3. 对于数字类型,也可使用Integer.toString()等特定类型的方法。这些方法简单高效,适用于不同场景。
43 7
|
1月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?