java stringbuffer原理_深入理解Java:String

在讲解String之前,我们先了解一下Java的内存结构。

一、Java内存模型

按照官方的说法:Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。

JVM主要管理两种类型内存:堆和非堆,堆内存(Heap Memory)是在 Java 虚拟机启动时创建,非堆内存(Non-heap Memory)是在JVM堆之外的内存。

简单来说,非堆包含方法区、JVM内部处理或优化所需的内存(如 JITCompiler,Just-in-time Compiler,即时编译后的代码缓存)、每个类结构(如运行时常数池、字段和方法数据)以及方法和构造方法的代码。

Java的堆是一个运行时数据区,类的(对象从中分配空间。这些对象通过new、newarray、 anewarray和multianewarray等指令建立,它们不需要程序代码来显式的释放。堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。

栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量数据(int, short, long, byte, float, double, boolean, char)和对象句柄(引用)。

虚拟机必须为每个被装载的类型维护一个常量池。常量池就是该类型所用到常量的一个有序集合,包括直接常量(string,integer和 floating point常量)和对其他类型,字段和方法的符号引用。

对于String常量,它的值是在常量池中的。而JVM中的常量池在内存当中是以表的形式存在的, 对于String类型,有一张固定长度的CONSTANT_String_info表用来存储文字字符串值,注意:该表只存储文字字符串值,不存储符号引用。说到这里,对常量池中的字符串值的存储位置应该有一个比较明了的理解了。在程序执行的时候,常量池会储存在Method Area,而不是堆中。常量池中保存着很多String对象; 并且可以被共享使用,因此它提高了效率

具体关于JVM和内存等知识请参考:

二、案例解析

64023455_1.gif

public static void main(String[] args) {

/**

* 情景一:字符串池

* JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象;

* 并且可以被共享使用,因此它提高了效率。

* 由于String类是final的,它的值一经创建就不可改变。

* 字符串池由String类维护,我们可以调用intern()方法来访问字符串池。*/

String s1 = "abc";

//↑ 在字符串池创建了一个对象

String s2 = "abc";

//↑ 字符串pool已经存在对象“abc”(共享),所以创建0个对象,累计创建一个对象

System.out.println("s1 == s2 :"+(s1==s2));

//↑ true 指向同一个对象,

System.out.println("s1.equals(s2) :" + (s1.equals(s2)));

//↑ true 值相等//↑------------------------------------------------------over

/**

* 情景二:关于new String("")

**/

String s3 = new String("abc");

//↑ 创建了两个对象,一个存放在字符串池中,一个存在与堆区中;//↑ 还有一个对象引用s3存放在栈中

String s4 = new String("abc");

//↑ 字符串池中已经存在“abc”对象,所以只在堆中创建了一个对象

System.out.println("s3 == s4 :"+(s3==s4));

//↑false s3和s4栈区的地址不同,指向堆区的不同地址;

System.out.println("s3.equals(s4) :"+(s3.equals(s4)));

//↑true s3和s4的值相同

System.out.println("s1 == s3 :"+(s1==s3));

//↑false 存放的地区多不同,一个栈区,一个堆区

System.out.println("s1.equals(s3) :"+(s1.equals(s3)));

//↑true 值相同//↑------------------------------------------------------over

/**

* 情景三:

* 由于常量的值在编译的时候就被确定(优化)了。

* 在这里,"ab"和"cd"都是常量,因此变量str3的值在编译时就可以确定。

* 这行代码编译后的效果等同于: String str3 = "abcd";*/

String str1 = "ab" + "cd"; //1个对象

String str11 = "abcd";

System.out.println("str1 = str11 :"+ (str1 == str11));

//↑------------------------------------------------------over

/**

* 情景四:

* 局部变量str2,str3存储的是存储两个拘留字符串对象(intern字符串对象)的地址。

*

* 第三行代码原理(str2+str3):

* 运行期JVM首先会在堆中创建一个StringBuilder类,

* 同时用str2指向的拘留字符串对象完成初始化,

* 然后调用append方法完成对str3所指向的拘留字符串的合并,

* 接着调用StringBuilder的toString()方法在堆中创建一个String对象,

* 最后将刚生成的String对象的堆地址存放在局部变量str3中。

*

* 而str5存储的是字符串池中"abcd"所对应的拘留字符串对象的地址。

* str4与str5地址当然不一样了。

*

* 内存中实际上有五个字符串对象:

* 三个拘留字符串对象、一个String对象和一个StringBuilder对象。*/

String str2 = "ab"; //1个对象

String str3 = "cd"; //1个对象

String str4 = str2+str3;

String str5 = "abcd";

System.out.println("str4 = str5 :" + (str4==str5)); //false//↑------------------------------------------------------over

/**

* 情景五:

* JAVA编译器对string + 基本类型/常量 是当成常量表达式直接求值来优化的。

* 运行期的两个string相加,会产生新的对象的,存储在堆(heap)中*/

String str6 = "b";

String str7 = "a" + str6;

String str67 = "ab";

System.out.println("str7 = str67 :"+ (str7 == str67));

//↑str6为变量,在运行期才会被解析。

final String str8 = "b";

String str9 = "a" + str8;

String str89 = "ab";

System.out.println("str9 = str89 :"+ (str9 == str89));

//↑str8为常量变量,编译期会被优化//↑------------------------------------------------------over

}

64023455_1.gif

总结:

1.String类初始化后是不可变的(immutable)

这一说又要说很多,大家只要知道String的实例一旦生成就不会再改变了,比如说:String str=”kv”+”ill”+” “+”ans”; 就是有4个字符串常量,首先”kv”和”ill”生成了”kvill”存在内存中,然后”kvill”又和” ” 生成 “kvill “存在内存中,最后又和生成了”kvill ans”;并把这个字符串的地址赋给了str,就是因为String的”不可变”产生了很多临时变量,这也就是为什么建议用StringBuffer的原 因了,因为StringBuffer是可改变的。

下面是一些String相关的常见问题:

String中的final用法和理解

final StringBuffer a = new StringBuffer("111");

final StringBuffer b = new StringBuffer("222");

a=b;//此句编译不通过  final StringBuffer a = new StringBuffer("111");

a.append("222");// 编译通过

可见,final只对引用的"值"(即内存地址)有效,它迫使引用只能指向初始指向的那个对象,改变它的指向会导致编译期错误。至于它所指向的对象的变化,final是不负责的。

2.代码中的字符串常量在编译的过程中收集并放在class文件的常量区中,如"123"、"123"+"456"等,含有变量的表达式不会收录,如"123"+a。

3.JVM在加载类的时候,根据常量区中的字符串生成常量池,每个字符序列如"123"会生成一个实例放在常量池里,这个实例是不在堆里的,也不会被GC,这个实例的value属性从源码的构造函数看应该是用new创建数组置入123的,所以按我的理解此时value存放的字符数组地址是在堆里,如果有误的话欢迎大家指正。

4.使用String不一定创建对象

在执行到双引号包含字符串的语句时,如String a = "123",JVM会先到常量池里查找,如果有的话返回常量池里的这个实例的引用,否则的话创建一个新实例并置入常量池里。如果是 String a = "123" + b (假设b是"456"),前半部分"123"还是走常量池的路线,但是这个+操作符其实是转换成[SringBuffer].Appad()来实现的,所以最终a得到是一个新的实例引用,而且a的value存放的是一个新申请的字符数组内存空间的地址(存放着"123456"),而此时"123456"在常量池中是未必存在的。

要注意: 我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的对象。只有通过new()方法才能保证每次都创建一个新的对象

5.使用new String,一定创建对象

在执行String a = new String("123")的时候,首先走常量池的路线取到一个实例的引用,然后在堆上创建一个新的String实例,走以下构造函数给value属性赋值,然后把实例引用赋值给a:

64023455_1.gif

public String(String original) {

int size = original.count;

char[] originalValue = original.value;

char[] v;

if (originalValue.length > size) {

//The array representing the String is bigger than the new//String itself. Perhaps this constructor is being called//in order to trim the baggage, so make a copy of the array.

int off = original.offset;

v = Arrays.copyOfRange(originalValue, off, off+size);

} else {

//The array representing the String is the same//size as the String, so no point in making a copy.

v = originalValue;

}

this.offset = 0;

this.count = size;

this.value = v;

}

64023455_1.gif

从中我们可以看到,虽然是新创建了一个String的实例,但是value是等于常量池中的实例的value,即是说没有new一个新的字符数组来存放"123"。

如果是String a = new String("123"+b)的情况,首先看回第4点,"123"+b得到一个实例后,再按上面的构造函数执行。

6.String.intern()

String对象的实例调用intern方法后,可以让JVM检查常量池,如果没有实例的value属性对应的字符串序列比如"123"(注意是检查字符串序列而不是检查实例本身),就将本实例放入常量池,如果有当前实例的value属性对应的字符串序列"123"在常量池中存在,则返回常量池中"123"对应的实例的引用而不是当前实例的引用,即使当前实例的value也是"123"。

public native String intern();

存在于.class文件中的常量池,在运行期被JVM装载,并且可以扩充。String的 intern()方法就是扩充常量池的 一个方法;当一个String实例str调用intern()方法时,Java 查找常量池中 是否有相同Unicode的字符串常量,如果有,则返回其的引用,如果没有,则在常 量池中增加一个Unicode等于str的字符串并返回它的引用;看示例就清楚了

64023455_1.gif

public static void main(String[] args) {

String s0 = "kvill";

String s1 = new String("kvill");

String s2 = new String("kvill");

System.out.println( s0 == s1 ); //false

System.out.println( "**********" );

s1.intern(); //虽然执行了s1.intern(),但它的返回值没有赋给s1

s2 = s2.intern(); //把常量池中"kvill"的引用赋给s2

System.out.println( s0 == s1); //flase

System.out.println( s0 == s1.intern() ); //true//说明s1.intern()返回的是常量池中"kvill"的引用

System.out.println( s0 == s2 ); //true

}

64023455_1.gif

最后我再破除一个错误的理解:有人说,“使用 String.intern() 方法则可以将一个 String 类的保存到一个全局 String 表中 ,如果具有相同值的 Unicode 字符串已经在这个表中,那么该方法返回表中已有字符串的地址,如果在表中没有相同值的字符串,则将自己的地址注册到表中”如果我把他说的这个全局的 String 表理解为常量池的话,他的最后一句话,”如果在表中没有相同值的字符串,则将自己的地址注册到表中”是错的:

64023455_1.gif

public static void main(String[] args) {

String s1 = new String("kvill");

String s2 = s1.intern();

System.out.println( s1 == s1.intern() ); //false

System.out.println( s1 + " " + s2 ); //kvill kvill

System.out.println( s2 == s1.intern() ); //true

}

64023455_1.gif

在这个类中我们没有声名一个”kvill”常量,所以常量池中一开始是没有”kvill”的,当我们调用s1.intern()后就在常量池中新添加了一 个”kvill”常量,原来的不在常量池中的”kvill”仍然存在,也就不是“将自己的地址注册到常量池中”了。

s1==s1.intern() 为false说明原来的”kvill”仍然存在;s2现在为常量池中”kvill”的地址,所以有s2==s1.intern()为true。

StringBuffer与StringBuilder的区别,它们的应用场景是什么?

jdk的实现中StringBuffer与StringBuilder都继承自AbstractStringBuilder,对于多线程的安全与非安全看到StringBuffer中方法前面的一堆synchronized就大概了解了。

这里随便讲讲AbstractStringBuilder的实现原理:我们知道使用StringBuffer等无非就是为了提高java中字符串连接的效率,因为直接使用+进行字符串连接的话,jvm会创建多个String对象,因此造成一定的开销。AbstractStringBuilder中采用一个char数组来保存需要append的字符串,char数组有一个初始大小,当append的字符串长度超过当前char数组容量时,则对char数组进行动态扩展,也即重新申请一段更大的内存空间,然后将当前char数组拷贝到新的位置,因为重新分配内存并拷贝的开销比较大,所以每次重新申请内存空间都是采用申请大于当前需要的内存空间的方式,这里是2倍

StringBuffer 始于 JDK 1.0

StringBuilder 始于 JDK 1.5

从 JDK 1.5 开始,带有字符串变量的连接操作(+),JVM 内部采用的是

StringBuilder 来实现的,而之前这个操作是采用 StringBuffer 实现的。

我们通过一个简单的程序来看其执行的流程:

64023455_1.gif

public class Buffer {

public static void main(String[] args) {

String s1 = "aaaaa";

String s2 = "bbbbb";

String r = null;

int i = 3694;

r = s1 + i + s2;

for(int j=0;i<10;j++){

r+="23124";

}

}

}

64023455_1.gif

使用命令javap -c Buffer查看其字节码实现:

64023455_2.png

将清单1和清单2对应起来看,清单2的字节码中ldc指令即从常量池中加载“aaaaa”字符串到栈顶,istore_1将“aaaaa”存到变量1中,后面的一样,sipush是将一个短整型常量值(-32768~32767)推送至栈顶,这里是常量“3694”,更多的Java指令集请查看另一篇文章“Java指令集”。

让我们直接看到13,13~17是new了一个StringBuffer对象并调用其初始化方法,20~21则是先通过aload_1将变量1压到栈顶,前面说过变量1放的就是字符串常量“aaaaa”,接着通过指令invokevirtual调用StringBuffer的append方法将“aaaaa”拼接起来,后续的24~30同理。最后在33调用StringBuffer的toString函数获得String结果并通过astore存到变量3中。

看到这里可能有人会说,“既然JVM内部采用了StringBuffer来连接字符串了,那么我们自己就不用用StringBuffer,直接用”+“就行了吧!“。是么?当然不是了。俗话说”存在既有它的理由”,让我们继续看后面的循环对应的字节码。

37~42都是进入for循环前的一些准备工作,37,38是将j置为1。44这里通过if_icmpge将j与10进行比较,如果j大于10则直接跳转到73,也即return语句退出函数;否则进入循环,也即47~66的字节码。这里我们只需看47到51就知道为什么我们要在代码中自己使用StringBuffer来处理字符串的连接了,因为每次执行“+”操作时jvm都要new一个StringBuffer对象来处理字符串的连接,这在涉及很多的字符串连接操作时开销会很大。

参考:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/566636.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java dubbo jsf_cubelink

cubelink概要设计[TOC]1. 撰写记录更新时间内容作者2017-08-23 08:39:31撰写参数回调章节内容林斌2017-08-22 21:26:52增加了异步响应和异步回调章节林斌2017-08-22 14:36:36确定文档结构和大致框架林斌2. 设计目标设计一个具备治理&#xff0c;监控&#xff0c;服务发现能力的…

用java输出图形_java基础-输出一个简单的图形。

最近写了几行代码输出下面的图形。不多说了,代码如下。 * *** ***** public class javaDemoOne {/** * @brief 主函数 * @author wks * @param args */ public static void main(String[] args) {// TODO Auto-generated method stub inputGraPh(); System.out.print("\n…

python 的案例实战_python案例实战之一

分析思路&#xff1a;1、明确分析目标&#xff1b;2、导入库、导入数据&#xff1b;3、简单查看下数据行列、整体情况&#xff1b;4、数据清洗&#xff1b;5、确定维度和指标&#xff1b;6、分析并作图1、查看整体数据情况1.1引入使用的库import numpy as npimport pandas as p…

django 模板mysql_59 Django基础三件套 , 模板{{}}语言 , 程序连mysql Django项目app Django中ORM的使用...

主要内容:https://www.cnblogs.com/liwenzhou/p/8688919.html1 form表单中提交数据的三要素a : form标签必须要有action和method的属性b : 所有获取用户输入的标签必须放在form表单里,也必须要有那么name属性.因为往后端提交数据的时候name所对应的是关键字, input输入的值为va…

java不大于6位_末尾带4的完全平方数的数量并且打印输出_Java计算一个数加上100是完全平方数,加上168还是完全平方数...

题目&#xff1a;一个整数&#xff0c;它加上100后是一个完全平方数&#xff0c;加上168又是一个完全平方数&#xff0c;请问该数是多少&#xff1f;程序分析&#xff1a;在10万以内判断&#xff0c;先将该数加上100后再开方&#xff0c;再将该数加上268后再开方&#xff0c;如…

java使用Encoding导什么包_String getEncoding()

String getEncoding()描述 (Description)java.io.OutputStreamWriter.getEncoding()方法返回此流使用的字符编码的名称。如果编码具有历史名称&#xff0c;则返回该名称; 否则返回编码的规范名称。如果使用OutputStreamWriter(OutputStream&#xff0c;String)构造函数创建此实…

JAVA中增强循环中用线程_在Java中以循环方式运行线程

我是Java中的多线程和同步的新手。我正在尝试实现一项任务&#xff0c;其中给了我5个文件&#xff0c;每个文件将由一个特定线程读取。每个线程应从文件读取一行&#xff0c;然后将执行转发到下一个线程&#xff0c;依此类推。当所有5个线程都读取第一行时&#xff0c;然后再次…

java数据结构期末复习_java数据结构复习02

1.递归问题1.1计算阶乘packageinterview.recursion;importjava.util.Scanner;public classFact {public static voidmain(String[] args) {System.out.println("请输入n的值&#xff1a;");Scanner in newScanner(System.in);int n in.nextInt();int num fact(n);Sys…

java中methods方法_java中Class.getMethod方法

Method Class.getMethod(String name, Class>... parameterTypes)的作用是获得对象所声明的公开方法该方法的第一个参数name是要获得方法的名字&#xff0c;第二个参数parameterTypes是按声明顺序标识该方法形参类型。person.getClass().getMethod("Speak", null)…

centos6 yum快速安装mysql_centos6.10 yum安装mysql 5.6-Go语言中文社区

一、检查系统是否安装其他版本的MYSQL数据#yum list installed | grep mysql#yum -y remove 文件名二、安装及配置# wget http://repo.mysql.com/mysql-community-release-el6-5.noarch.rpm# rpm -ivh mysql-community-release-el6-5.noarch.rpm# yum repolist all | grep mysq…

java struts1_struts1.x

struts1.x摘要: 要想使用Struts&#xff0c;至少要依靠两个配置文件&#xff1a;web.xml和struts-config.xml。其中web.xml用来安装Struts框架。而struts-config.xml用来配置在Struts框架中要使用的资源。如Formbean、Action、插件等。如果使用了某些插件&#xff0c;如Validat…

java加花免杀_UPX加壳免杀添加花指令

UPX加壳、免杀、添加花指令是一款用于制作免杀的给力的工具。使用该软件可以进行软件的UPX加壳、E语言免杀、添加花指令。如果你正在为自己制作的软件过不了杀毒&#xff0c;那么赶快使用这款神器吧。加花方法&#xff1a;1.直接加花记住入口点---找零区域---NOP填充---记住新入…

java scanner转string,Java InputStream to String 转化

1. 概况这篇文章主要是讲怎样将InputStream转换为String。采用[weblink url"http://code.google.com/p/guava-libraries/"]Guava[/weblink]、[weblink url"http://commons.apache.org/proper/commons-io/"]Apache Commons IO [/weblink]以及普通Java代码实…

kafka php 教程,php的kafka踩坑(一)

最近项目上有一个需要用到消息队列的功能&#xff0c;从网上找了一些php相关的kafka使用的教程和博客&#xff0c;大抵都是安装php的拓展librdkafka(这里就不讲这个拓展的安装方法了&#xff0c;搜一下还是有很多教程的)&#xff0c;然后直接用这个拓展进行开发&#xff0c;但是…

二叉树两节点距离java,求二叉树中两个节点的最远距离

问题定义如果我们把二叉树看成一个图&#xff0c;父子节点之间的连线看成是双向的&#xff0c;我们姑且定义"距离"为两节点之间边的个数。写一个程序求一棵二叉树中相距最远的两个节点之间的距离。计算一个二叉树的最大距离有两个情况:情况A: 路径经过左子树的最深节…

php中update()函数,update_option()函数

update_option()函数的功能是更新wp_options表中指定的一条数据。可以使用此函数代替add_option函数&#xff0c;尽管它不够灵活。 update_option函数会检查并判断选项是否已经存在。如果不存在&#xff0c;用add_option (’option_name’, ‘option_value’)添加选项。除非用户…

java解析MT940报文,swift MT报文解析处理

swift 官方资料&#xff1a;https://www2.swift.com/knowledgecentre/publications/us5mc_20180720/2.0?topicalec.htm#genalecswift 百科&#xff1a;https://baike.baidu.com/item/SWIFT/1108075prowide - swift 报文处理 开源框架&#xff1a;https://www.prowidesoftware.…

php怎么实现匿名评论,PHP-匿名对象与匿名类的实现过程-0905

* 匿名类:* 1. php 7.0 才支持* 2. 类似于匿名函数,就是没有名称的类* 3. 匿名类适合于一次性的创建与引用* 4. 匿名类总是与: new 配套使用类的三种访问方式实例/*** 匿名类:* 1. php 7.0 才支持* 2. 类似于匿名函数,就是没有名称的类* 3. 匿名类适合于一次性的创建与引用* 4.…

php cli 编程,php-cli下编程如何分层架构、面向对象、统一入口文件?

以往写cli下运行的业务或者测试代码&#xff0c;总是新建文件&#xff0c;面向过程编写代码。几次之后&#xff0c;cli目录下好多文件&#xff0c;即便勉强在一个cli测试文件中写了一个类&#xff0c;也是让其中的一个方法自启动&#xff0c;要测试别的方法&#xff0c;总是要修…

matlab中平均函数用法,matlab中怎样在X的指定范围内求y的平均值

有两组数据&#xff0c;前面一组值设为x后面一组设为y。x是坐标的变化范围&#xff0c;y是每个坐标下力的大小&#xff0c;怎样在X的指定范围内求y的平均值&#xff1f;&#xff1f;比如下面x范围是从-18.19959641到-18.00003052之内的 怎样求得-18.19959641到-18.18049049这个…