java String、常量池、StringBuffer总结

String和常量池

常量池(constant pool)指的是在编译期被确定,并被保存在已编译的.class文件中的一些数据。它包括了关于类、方法、接口等中的常量,也包括字符串常量(字符串常量池)。
事例1如下:
	public static void s1(){
		String s0 = "abc"; //常量池中不存在"abc",在常量池中创建"abc"并返回引用
		String s1 = "abc"; //常量池中存在"abc",返回"abc"的引用付给s1
		String s2 = new String("abc"); //"abc"存储在堆上,s2引用地址存储在栈上
		
		System.out.println(s0 == s1); //=>true
		System.out.println(s0 == s2); //=>false
	}

在上例中"abc"就是字符串常量,在编译期就可以确定,会存储在常量池中。常量池确保一个字符串只有一个拷贝。s0和s1都指向常量池中的"abc"所以返回true。s2运行期创建,指向堆中的"abc",所以s0 == s2为fasle。

事例2如下:
	public static void s2(){
		String c="c";
		final String cFinal = "c";
		
		String s1 = "abc";
		String s2 = "ab"+c;
		String s3 = "a"+"b"+"c";
		String s4 = "ab"+cFinal;
		
		System.out.println(s1 == s2); //=>false
		System.out.println(s1 == s3); //=>true
		System.out.println(s1 == s4); //=>true
		
	}

s1指向常量池中的"abc"。
s2在编译器无法确定值,因为使用"+"连接字符串时有一个对字符串的引用存在,无法再编译期确定值,只能在运行期确定值。s1 == s2 =>false
s3使用"+"连接了三个字符串常量(“a”、"b"、"c"),在编译期就可以确定,最终产生一个"abc"常量。s1 == s3 =>true
s4使用"+"连接字符串时也有一个外来的对字符串的引用c,但是这个引用是final的,就是说第一次初始化之后是不也许更改的。等同于s3。s1 == s3 =>true
我们可以这样认为:
在使用 “+”连接字符串时如果连接的都是字符串常量,那么最终也会产生一个字符串常量(例如s2)放入连接池。若有外来对字符的引用,将不会产生一个字符串常量放入常量池中。除非这个外来的字符串的引用是用fianl修饰的。s1 == s3  =>true
注意哟,如果放入的字符已经存在常量池中,会直接返回已存在字符的引用。

实例3如下:
	public static void s3(){
		String s1 = "abc";
		String s2 = new String("abc");
		
		System.out.println(s1 == s2.intern()); //=>true
	}

我们在这里要说的是String类的intern()方法。intern()是扩充常量池的一个方法。一个String实例调用intern()会将这个实例的值放入常量池中,如果常量池已经存在该字符串值就直接返回已存在字符串的引用。
s1 == s2.intern()就说明了s2.intern()返回的是常量池中"abc"的引用。

字符串是不可变的

java中String类不属于基本类型,属于引用类型,是一个中特殊的对象。说字符串是不可变的,是因为每次对字符的操作都会产生一个新的字符串。
待续。。。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值