java中String s="abc"及String s=new String("abc")详解

本文转自:https://www.cnblogs.com/quyanhui/p/3781663.html
本人书写这个纯粹只是为了便于自己查找,开放是为了方便各位指出我的错误。

1、java中的堆与栈都是Java中用来RAM内存中使用的,Java自动通过GC自动管理堆栈,无需程序员管理。
2、存储在栈的对象,变量,生命周期都是已知的程序结束对象就会被死亡,优势就是存取速度快。但相比于堆,它可存储数据的大小,生命周期都是固定的,不够灵活。 另外,栈的数据可以共享。堆的优势便是可以动态的分配大小,生存周期也不需要告诉编译器(但是容易引起GC问题),GC会自动处理这个。但是缺点就是,由于是在运行中动态的分配内存,所以存取速度会慢一些。

==是判断两个对象是否是同一个对象 
equals是进行值的判断 
String   a   =   new   String( "aaa "); 
String   b   =   new   String( "a "); 
b   +=   "aa "; 
则   a==b   //错误 
      a.equals(b)//正确

除了String和封装器,equals()和“==”没什么区别 
但String和封装器重写了equals(),所以在这里面,equals()指比较字符串或封装对象对应的原始值是否相等, "== "是比较两个对象是否为同一个对象

首先我们看一下Java中变量的的语义:
Java中的语义定义有两种,原始类型的定义是值(value)定义,也就是说,你给一个原始类型赋值,就改变了这个数据本身。第二种对象类型,对象类型的变量是引用语义,也就是说你给一个对象类型赋值只是让它指向了一个对象(对象地址or地址),但是不会改变原来引用对象的值。

《String的特性》
1、String类是final的,不可被继承的。
2、String类本质是字符数组char[],并且其值不可变。
3、String类对象有个特殊的创建方式,就是直接指定String s = “abc”,“abc”就代表一个字符串对象,而s就是“abc”的地址(我理解为s指向“abc”的地址),也叫做“abc”对象的引用。
4、String对象可以通过“+”串联,串联后会形成新的字符串对象。
5、String运行时会维护一个String Pool(String池),JavaDoc翻译很模糊“字符串缓冲区”。String池用来存放运行时产生的各种字符串,并且池中的字符串不可以重复。而一般对象不存在这个缓存区,并且创建的对象仅仅存在于方法的堆栈中。
6、创建字符串的方式很多,归纳起来分为三种:
1)使用new关键字创建,例如:String s = new String(“abc”);(注意这里是创建了两个对象)
2)直接指定 s2 = “abc”;
3)使用串联生成新的字符串。例如 String s3 = “ab” + “cd”;

《String对象的创建》
String对象的创建也有很多门道,关键就是明白其中的原理。
原理1:当使用任何方式来创建一个字符串对象s=X的时候,Java运行时(运行中JVM)会拿着这个X去String池中查找是否存在相同的对象,如果不存在,则在池中添加,如果存在,怎不添加。

原理2;Java中,只要使用new关键字创建对象,则一定会(在堆中或者栈中)创建一个新的对象。

原理3:直接指定或者使用纯字符串串联来创建对象,则仅仅检查维护Stirng池中字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。

原理4:使用包含变量的方式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区中创建新的字符串对象。

《不可变类》
Java为了提高效率,对String类型进行特殊处理–为String类型提供了字符串池
定义一个String类型的变量有两种方式:
String s = “tom”;
String s = new(“tom”);

如果你使用第一种方式,那么当你申明一个内容是“tom”的String时,他将会使用串池里原来的那个内存(地址),而不会重新分配内存,也就是说String s = “tom“;将会指向同一块内存。而如果使用第二种方式,不管串池里面有没有,都会在堆中开辟一块新的内存,创建一个新的对象。

另外关于String类型是不可改变的问题:String类型是不可改变的,也就是说,当你想改变一个String对象的时候,例如Stirng s = “tom”;那么虚拟机不会改变原来的对象,而是会生成一个新的String对象,然后让s取直向它,如果原来的那个“tom”没有任何对象去引用它,GC将会回收它。

最后,关于String堆栈的问题
String s = new String();分析堆栈,是先定义s,还是先new String()。
1、String str1 = “abc”;
System.out.println(str1 == “abc”);
步骤:
1)栈中开辟一块空间存放引用str1;
2)String池中开辟一块空间,存放String常量”abc”;
3)引用str1指向池中String常量”abc”。
4)str1所指代的地址即常量”abc”所在地址,输出为true。

2、String str2 = new String(“abc”);
System.out.println(str2 == “abc”);\
步骤:
1)栈中开辟一块空间存放引用str2。
2)堆中开辟一块空间存放一个新建的String对象”abc”。
3)引用str2指向堆中新创建的对象”abc”。
4)str2所指代的对象地址为堆中地址,而常量”abc”地址在池中,输出为false。

3.Sting str3 = new String(“abc”);
System.out.println(str3 == str2);
步骤:
1)栈中开辟一块空间存放引用str3。
2)堆中开辟一块新空间存放另外一个(不同于str2所指)新建的String对象。
3)引用str3指向另外新建的那个String对象。
4)str2和str3分别指向堆中不同的String对象,地址也不相同,输出为false。

4.String str4 = “a” + “b”;
System.out.println(str4 == “ab”);
步骤:
1)栈中开辟一块空间存放引用str4。
2)根据编译器合并已知量的优化功能,池中开辟一块空间,存放合并后的String常量”ab”。
3)引用str4指向池中的常量”ab”。
4)str4所指即池中常量”ab”,输出为true。

5.final String s = “a”;//注意这个s用final修饰,相当于常量。
String str5 = s +”b”;
System.out.println(str5 == “ab”);
步骤:
同4。

6.String s1 = “a”;
String s2 = “b”;
String str6 = s1 + s2;
System.out.println(str6 == “ab”);
步骤:
1)栈中开辟一块空间存放引用s1,s1指向池中String常量”a”,
2)栈中开辟一块空间存放引用s2,s2指向池中String常量”b”,
3)栈中开辟一块空间存放引用str6,
4)s1+s2通过StringBuilder的最后一步toString()方法还原一个新的String对象”ab”,因此堆中开辟一块空间存放对象。
5)引用str6指向堆中(s1 + s2)所还原的新String对象
6)str6指向的对象在堆中,而常量”ab”在池中,所有输出false。

7.String str7= “abc”.substring(0, 2);
步骤:
1)栈中开辟一块空间存放引用str7,
2)substring()方法还原一个新的String对象”ab”(不同于str6所指),堆中开辟一块空间存放此对象。
3)引用str7指向堆中新创建的String对象

8.String str8 = “abc”.toUpperCase();
步骤:
1)栈中开辟一块空间存放引用str8,
2)toUpperCase()方法还原一个新的String对象”ABC”,池中并未开辟新的空间存放String常量”ABC”
3)引用str8指向堆中的新String对象

9.String s = “abc”;
String s1 = s;
System.out.println(s1 == “abc”);
s = s + “hello”;
System.out.println(s1==”abc”);
System.out.println(s==”abc”);
步骤:
1)栈中开辟一块空间存放s;
2)String池中开辟一块空间用于存放”abc”;栈中开辟一块空间存放变量s1;
3)系统输出true,,在堆中开辟一块空间用于存放”abchello”;
4)引用s指向堆中”abchello”;
5)系统输出true,然后在输出false。

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值