本文主要解决以下几个问题

  • String源码解析?
  • String和new String的区别?
  • String通过“+”或concat累加时的对象创建机制?
  • StringBuilder和StringBuffer?区别和联系?

String源码解析

String类的定义

public final class String
implements java.io.Serializable, Comparable<String>, CharSequence { ...... }

通过上面的代码可以得出以下结论:

  • String类使用final关键字修饰,表示String类是最终类,不可以被继承;
  • String类是CharSequence的一个实现类;
  • String类实现了Serializable接口,表示String类可以被序列化;
  • String类实现了Comparable接口,表示String类对象之间可以直接进行比较

String#value

/** The value is used for character storage. */
private final char value[];
  • String中的value属性用于字符存储;
  • String中的value属性使用final关键字修饰,表示value不可变

由于value不可变,因此,String对象具有以下两个特点:

  • String对象地址不可变(Immutable)
  • String对象是线程同步的(can be shared)

String#concat()

public String concat(String str) {
int otherLen = str.length();
if (otherLen == 0) {
return this;
}
int len = value.length;
char buf[] = Arrays.copyOf(value, len + otherLen);
str.getChars(buf, len);
return new String(buf, true);
}

不管上面对字符串进行了什么处理,只看最后一行,返回了一个new String,因此,concat()方法实际上是返回了一个新的字符串。也就是说,每次调用concat()方法时,concat()方法中的参数会生成一个新的字符串变量,执行后得到的新字符串又是一个新的字符串变量。使用“+”进行字符串累加时也是一样的机制。

String和new String的区别

Java虚拟机中用于存储的区域分为栈、堆和方法区,方法区中包括另一片区域,称为常量池。String的存储主要涉及到的存储区域就是堆、栈和常量池。

栈中主要存储的是简单类型的变量和引用类型变量的存储地址;堆中存储的是引用类型数据本身;常量池中存储的是常量。

String变量的声明和定义方式有两种:

  • String str1 = new String("mystr");
  • String str2 = "mystr";

以上两种创建String对象的方式的区别是:

  • 运行时机的不同:第一种方式中有new关键字,是在程序运行期间才会执行的;而第二种方式是在编译期间就执行了;
  • 创建对象过程的不同:使用第一种方式创建String对象时,首先在堆中创建一个String对象,然后在栈中开辟一片空间,指向堆中的这一片区域,最后去常量池中寻找是否存在这个字符串常量,如果不存在,则在常量池中创建一个;使用第二种方式创建String对象时,会直接去常量池中寻找是否有这个字符串常量,如果有则直接引用到栈中,如果没有则先创建一个再引用到栈中。
  • 通过上一点可以得出结论:通过第一种方式创建String对象之后,在堆中、常量池中分别有一个对象,因此是创建了一个或两个对象;通过第二种方式创建String对象之后,在常量池中有一个对象,因此是创建了零个或一个对象;
  • 使用第一种方式,栈中的指针指向的是堆中的对象;使用第二种方式,栈中的指针指向的是常量池中的对象

StringBuilder和StringBuffer

前面说到,使用“+”和concat()方法对字符串进行拼接的时候,会创建额外的字符串变量,因此,当我们需要进行字符串拼接的时候,比较提倡的方法是使用StringBuilder或StringBuffer。

StringBuilder和StringBuffer中都提供了append()方法,可以实现字符串的拼接。

public final class StringBuffer
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence
public final class StringBuilder
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence

StringBuilder和StringBuffer都继承自AbstractStringBuilder,其底层依然是一个char数组,但不是final类型的。

// StringBuffer # append()
public synchronized StringBuffer append(CharSequence s) {
toStringCache = null;
super.append(s);
return this;
}
// StringBuilder # append()
public StringBuilder append(CharSequence s) {
super.append(s);
return this;
}

从上面的StringBuffer和StringBuilder的append()方法的源码比较中可以发现,两个类中append()方法唯一的区别就是,StringBuffer中的append()方法添加了synchronized关键字,表示这个方法是线程同步的。

通过上面的分析得到结论:StringBuffer是线程安全的,而StringBuilder是线程不安全的。

由于StringBuilder是线程不安全的,因此StringBuilder处理字符串的性能要比StringBuffer高。

【JAVA - 基础】之String存储机制浅析的更多相关文章

  1. Java基础之String、StringBuffer、StringBuilder浅析

    Java基础之String.StringBuffer.StringBuilder浅析 一.前言: 位于java.lang包下的String.StringBuilder.StringBuffer一般都是 ...

  2. Java基础之多态和泛型浅析

    Java基础之多态和泛型浅析 一.前言: 楼主看了许多资料后,算是对多态和泛型有了一些浅显的理解,这里做一简单总结 二.什么是多态? 多态(Polymorphism)按字面的意思就是“多种状态”.在面 ...

  3. Java基础之Collection与Collections浅析

    Java基础之Collection与Collections浅析 一.前言: 位于Java.util包下的Collection与Collections都是Java中重要的工具类,它们都是Java集合框架 ...

  4. Java基础篇(JVM)——类加载机制

    这是Java基础篇(JVM)的第二篇文章,紧接着上一篇字节码详解,这篇我们来详解Java的类加载机制,也就是如何把字节码代表的类信息加载进入内存中. 我们知道,不管是根据类新建对象,还是直接使用类变量 ...

  5. 【Java基础】String 相关知识点总结

    String 相关知识点总结 字符串的不可变性 概述 String 被声明为 final,因此它不可继承 在 Java8 中,String 内部使用 char 数组存储数据 public final ...

  6. Java基础之String中equals,声明方式,等大总结

    无论你是一个编程新手还是老手,提到String你肯定感觉特别熟悉,因为String类我们在学习java基础的时候就已经学过,但是String类型有我们想象的那么简单吗?其实不然,String类型的知识 ...

  7. java基础之 垃圾回收机制

    1. 垃圾回收的意义 在C++中,对象所占的内存在程序结束运行之前一直被占用,在明确释放之前不能分配给其它对象:而在Java中,当没有对象引用指向原先分配给某个对象的内存时,该内存便成为垃圾.JVM的 ...

  8. Java 基础之 String 类

    String String 被声明为 final,因此不能被继承.(Integer 等包装类也不能被继承) 在 java8 中,String 内部使用 char 数组 来存储数据 public fin ...

  9. Java基础-String 存储机制管理

    JVM运行的时候,将内存分为两个部分,一部分是堆,一部分是栈.堆中存放的是创建对象,而栈中存放的则是方法调用过程中的局部变量或引用.在设计JAVA字符串对象内存实现的时候,在堆中又开辟了一块很小的内存 ...

随机推荐

  1. phpStorm //todo 的用途

    用phpstorm看到别人的代码使用了注释//todo,且todo是彩色的 我想这个应该是有点用的吧,于是百度了下,大概是可能由于某些原因,导致部分代码没有写.但又怕忘了, 用//todo就可以做提示 ...

  2. 《JavaScript设计模式与开发实践》-- 策略模式

    详情个人博客:https://shengchangwei.github.io/js-shejimoshi-celue/ 策略模式 1.定义 策略模式:定义一系列的算法,把它们一个个封装起来,并且使它们 ...

  3. mha格式的CT体数据转为jpg切片

    mha格式的CT体数据转为jpg切片 mha格式 .mha文件是一种体数据的存储格式,由一个描述数据的头和数据组成,一般我们拿到的原始医学影像的数据是.dcm也就是dicom文件,dicom文件很复杂 ...

  4. DataGridView内容居中显示

    DataGridView1.ColumnHeadersDefaultCellStyle.Alignment = DataGridViewContentAlignment.MiddleCenter '标 ...

  5. 学习笔记02form

    1.<form>为表单标签*如果要把数据提交到服务器,则需要将<input> <textarea> <select>等表单元素放到<from> ...

  6. 相关推导式-Python

    列表.’字典等推导式 #利用zip()函数同时给多个变量赋值 a = [1,2,3,4,5] b = [4,5,6,7,8] c = [9,2,3,4,0] l = [[1,2],[3,4]] for ...

  7. 零基础Linux入门之《Linux就该这么学》

    本书是由全国多名红帽架构师(RHCA)基于最新Linux系统共同编写的高质量Linux技术自学教程,极其适合用于Linux技术入门教程或讲课辅助教材,目前是国内最值得去读的Linux教材,也是最有价值 ...

  8. 5种常见Bean映射工具的性能比对

    本文由 JavaGuide 翻译自 https://www.baeldung.com/java-performance-mapping-frameworks .转载请注明原文地址以及翻译作者. 1. ...

  9. python协程总结

    概述 python多线程中因为有GIL(Global Interpreter Lock 全局解释器锁 )的存在,所以对CPU密集型程序显得很鸡肋:但对IO密集型的程序,GIL会在调用IO操作前释放,所 ...

  10. F#周报2019年第46期

    新闻 使用Pulumi和.NET Core创建现代云应用 宣告.NET Core 3.1预览版3 ML.NET模型构建器升级 .NET Framework修复工具 Mac上的Visual Studio ...