博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
String不可变类
阅读量:6212 次
发布时间:2019-06-21

本文共 3611 字,大约阅读时间需要 12 分钟。

  hot3.png

String,StringBuffer, StringBuilder 的区别是什么?String为什么是不可变的?

(1)String不可变,一旦String对象在内存中被创建,就不能被改变,所有修改String值的方法,并不是改变String对象本身,而是创建一个新String对象。(concat、substring中都会new String(),对象属性一旦初始化就不能改变

private final char value[];

(2)StringBuilder、StringBuffer可变性,是用char[],String实例可被改变。

(3)StringBuffer是线程安全的,而StringBuilder是非线程安全的。

  ps:线程安全会带来额外的系统开销,所以StringBuilder的效率比StringBuffer高

//1.final类,不能被继承

//2.构造String

//2.1调用构造函数创建String对象,一定产生新的空间,堆

//2.2直接赋值,字符串常量/字符串字面量,不一定产生新的空间,享元模式

String s1 = new String("world");String s2 = "hello";String s3 = "hello";System.out.println(s2==s3);//true/** 认为有一个字符串池的内存空间,里面放字符串常量* 每次,总是先从字符串池中找到该常量是否存在,* 如果有直接引用,如果没有,则创建然后引用*///3.字符串连接问题String s4 = "a"+"b"+"c";//直接等于"abc",jvm自动优化String s5 = "abc";System.out.println(s4==s5);//trueString s6 = 1+2+3+"abc";//6abcSystem.out.println(s6);String s7 = "abc"+1+2+3;//abc123System.out.println(s7);String s8 = "hello"+"world";String s9 = "helloworld";System.out.println(s8==s9);//trueString s10 = s3+"world";System.out.println(s9==s10);//new StringBuilder(s4).append("world");/** 4.StringBuffer、StringBuilder字符串拼接时用这两个类,效率高、节约内存,如果用字符串+* 性能差,并且浪费很多空间,产生很垃圾* +=new StringBuilder("s").append("world");* StringBuffer线程安全、StringBuilder线程不安全,在多线程的情况下比StringBuffer性能高* 单线程情况下同等*/public static void main(String[] args) {String a = getString(3);System.out.println(a);}public static String getString(int length){StringBuilder s = new StringBuilder();s.append("(");for(int i = 0;i

1 看下面一段代码.

String s = new String("abc");

String s1="abc";

String s2 = new String("abc");

System.out.println(s==s1);

System.out.println(s==s2);

System.out.println(s1==s2);

请问 前面三条语句分别创建了几个对象,分别是什么.后面的输出分别是什么

(1)String s = new String("abc"); 这句,创建了两个对象..其内容都是"abc".注意,s不是对象,只是引用.只有new生成的才是对象.

创建的流程是,首先括号里的"abc"先到String pool里看有没"abc"这个对象,没有则在pool里创建这个对象..所以这里就在pool创建了一个"abc"对象.然后 通过new语句又创建了一个"abc"对象..而这个对象是放在内存的堆里. .这里的s指向堆里的对象.

(2) String s1 = "abc"; 这条语句,s1当然还是引用.没啥可说的.后面的"abc".其实就是上面括号里的"abc".执行的是相同的操作.即 在pool里查找有没"abc"这个对象.没有则创建一个...很显然,第一条语句在pool里已经创建了一个"abc".所以这条语句没有创建对象,s1指向的是pool中的"abc"

(3)String s2 = new String("abc"); 这条语句,其实和第一条是一样的,但是,因为第一条已经在pool中创建了"abc"这个对象,所以,这条语句创建了一个对象.s2指向的是堆里的"abc".注意,虽然内容都是"abc",s与s2表示的是不同的对象

(4)接下来就很好说了.下面的三个==判断.(注意,==永远是判断内存地址是否相等) s与s1,一个指向堆里的对象,一个指向pool里的.很明显是不同的对象.s与s2.上面说了,虽然都是指向堆里的对象,内容也是"abc",但是也不是相同的对象.s1与s2.一个指向pool,一个指向堆.也不是相同的对象.所以三个都返回false.

2 第二个问题

String s = new String("abc");

String s1="abc";

String s2 = new String("abc");

System.out.println(s==s.intern());
System.
out.println(s1==s1.intern());

System.out.println(s.intern()==s2.intern());//s.intern 指向String pool里的地址

求最后输出是什么

解答.最后的答案是 false true true

intern()方法.按照jdk的帮助文档来说,是返回字符串对象的规范化表示形式。通俗一点说,就是返回对应这个字符串内容的那个pool里的对象.这样说也许还看不太明白,那可以拿具体例子来说

s1.intern().他的执行流程是,在pool里去查找s1对应的内容(也就是"abc").如果找到,则返回pool里的对象.如果没有(老实说,我没想到有哪种情况是没有的),则在Pool创建这个对象,并返回...

这样就很容易理解了.s1.intern返回的是pool里的"abc"对象.与s这个堆里的对象肯定不同,返回false.同理,s与s2.intern()也肯定不同,返回false.第三个,s1与s2.intern().其中s2.intern()返回的是pool中的"abc"对象,而s1也是指向pool中的"abc"对象.所以返回的是true:

3第三个问题

String hello = "hello";

String hel = "hel";
String
lo = "lo";
System.
out.println(hello == "hel" + "lo");

System.out.println(hello == "hel" + lo);

求输出的结果

解答 true false

首先,上面已经说明了,hello hel lo这三个都是指向pool中的对象..

现在我们考虑"hel" + "lo" 按照内容来说,两个相加也就是"hello".这个时候,这个会返回pool中的"hello"对象.所以,hello == "hel" + "lo" 返回的是true .

而"hel" + lo 虽然内容也是"hello",但是它将在堆里面生成一个"hello"对象,并返回这个对象...所以这里的结果是false

总结一下就是,如果加号两边的是字面值(字面值就是直接在""里写的值,比如上面的"hel"与"lo"),那么将在pool里查找有没对应内容的对象(这里的内容就是"hello"),并返回pool里的对象.这和hello是一样的....

如果加号两边不满足上面的条件(比如,两边的值是引用值或者堆里的字符串对象).那么将不会再pool里查找"hello",而是直接在堆里生成一个新的对象...

转载于:https://my.oschina.net/elain/blog/385441

你可能感兴趣的文章
我的友情链接
查看>>
长用小功能
查看>>
RHEl 5服务篇—部署Postfix邮件服务(二)配置邮件服务器的收、发信服务
查看>>
AD RMS 自我排错分析篇
查看>>
Kubernetes Deployment滚动更新场景分析
查看>>
多线程 3道题目 一
查看>>
我的友情链接
查看>>
c# 二维数组
查看>>
解决mx:AdvancedDataGrid表格中向上滚动导致数据混乱的非主流方法
查看>>
Buffer Cache内容强制写出到数据文件 alter system flush buffer_cache
查看>>
MicropPython的学习,如何从0到1?
查看>>
DNS(BIND) 服务器主从,实现高效率域名解析
查看>>
PHP 字符串变量
查看>>
Nginx+Lua环境安装
查看>>
Linuxshell快捷键
查看>>
three js相关的文档
查看>>
PHP获取和操作配置文件php.ini的几个函数
查看>>
83.LAMP设置默认主机
查看>>
sql笔记(来自oracle 的scott用户)
查看>>
bash_profile和.bashrc
查看>>