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

  • 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. 虚拟环境:virtualenv与virtualenvwrapper

    前言: 在使用 Python 开发的过程中,工程一多,难免会碰到不同的工程依赖不同版本的库的问题: 亦或者是在开发过程中不想让物理环境里充斥各种各样的库,引发未来的依赖灾难. 此时,我们需要对于不同的 ...

  2. ProvisionedAppxPackage VS AppxPackage

    正文 先来说说问题的由来. 在 Preinstall 的 component 中,有一支 component 叫做 MS_StartApp,这个 component 的行为是在预安装时为目标机器装入一 ...

  3. 还不会用FindBugs?你的代码质量很可能令人堪忧

    前言 项目中代码质量,往往需要比较有经验的程序员的审查来保证.但是随着项目越来越大,代码审查会变得越来越复杂,需要耗费越来越多的人力.而且程序员的经验和精力都是有限的,能审查出问题必定有限.而在对代码 ...

  4. 1、Hibernate-入门

    一.概述 1.什么是Hibernate: Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的orm框架,h ...

  5. AtCoder Grand Contest 038

    目录 \(\bf A - 01 \ Matrix\) \(\bf B- Sorting \ a \ Segment\) \(\bf C-LCMs\) \(\bf D-Unique \ Path\) 这 ...

  6. NOIP模拟 6

    考试时看了看T1,觉得是结论题,推了推没有成果,跑去看第二题, 题意很明确,求过定点的最小环,还没思考解题策略,然后觉得是水题 打了个tarjan找边双(据说会炸但是平均表现良好),在边双里暴力拆边找 ...

  7. MapReduce任务提交源码分析

    为了测试MapReduce提交的详细流程.需要在提交这一步打上断点: F7进入方法: 进入submit方法: 注意这个connect方法,它在连接谁呢?我们知道,Driver是作为客户端存在的,那么客 ...

  8. 腾讯开源进入爆发期,Plato助推十亿级节点图计算进入分钟级时代

    腾讯开源再次迎来重磅项目,14日,腾讯正式宣布开源高性能图计算框架Plato,这是在短短一周之内,开源的第五个重大项目. 相对于目前全球范围内其它的图计算框架,Plato可满足十亿级节点的超大规模图计 ...

  9. 花一天时间试玩vsphere6.7(EXSI)服务器版的vmware

    花一天时间试玩vsphere6.7(EXSI)服务器版的vmware 要注册账号(2019年11月14注册): 登陆网址:https://my.vmware.com/cn/group/vmware/h ...

  10. nyoj 208 + poj 1456 Supermarket (贪心)

    Supermarket 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 A supermarket has a set Prod of products on sal ...