1:创建一个空字符串对象,
2:创建一个字符串为空的字符串对象。
3:声明一个字符串对象,但并没有分配内存,而1,2已经分配了内存
对于最后一种表示,你不能if(abc==null),或者int length = abc.length();编译的时候会提示可能没有初始化.
String str="aaa"; //于栈上分配内存
String str=new String("aaa"); //于堆上分配内存
声明了一个string a;变量
在以后的判断中,a==""和a==null有何不同?
答:
如果没有给a赋过值,a==""会导致异常。
在实际处理时,往往认为""和null代表相同的含义,即都代表无值。
此时建议用如下语法:
if(a==null || a=="")
{
}
如果a为null,就不会执行后面的判断,直接返回true。
null是用来判断引用类型是否分配了存储空间
""是针对字符串的;
string类型实际上是字符串指针,也即是一个引用类型
所以如果没有给a赋过值,a==""会导致异常
所以if(a==null || a==""){}这种写法也是正确的
String s;在什么情况下可以等同于String s=null;而在什么情况下又不等同?!
考虑下面的代码:
//StringTest.java
public class StringTest {
static String s; //*
public static void main(String[] args) {
//String s; //**
System.out.println(s);
}
}
编译并运行上面的代码,将打印null。
可见标有*号的行是自动初始化了的(s被自动初始化为null)。
而如果把标有**号的行取消注释,代码将不能通过编译,这是因为这行定义的是本地变量,而本地变量是不会自动初始化的。
由此得出结论:
在成员变量的定义中,String s;等同于String s=null;
而在本地变量(方法变量)的定义中,String s;不等同于String s=null;,这时要使用s必须显式地赋值。
这些虽然是小知识点,但在实际应用中很重要,也很容易被一些人忽视,特此提出。
还有一点要说明的是:
只要是在方法在中定义变量都要显示赋初值,main()方法也不例外,而在方法之外编译器回自动赋初值。
用”ab”.intern()方法的时候会返回”ab”,但是这个方法会首先检查字符串池中是否有”ab”这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。
尽管在输出中调用intern方法并没有什么效果,但是实际上后台这个方法会做一系列的动作和操作。在调用”ab”.intern()方法的时候会返回”ab”,但是这个方法会首先检查字符串池中是否有”ab”这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。
可以看下面一个范例:
1 String str1 = " a " ;
2 String str2 = " b " ;
3 String str3 = " ab " ;
4 String str4 = str1 + str2;
5 String str5 = new String( " ab " );
6
7 System.out.println(str5.equals(str3));
8 System.out.println(str5 == str3);
9 System.out.println(str5.intern() == str3);
10 System.out.println(str5.intern() == str4);
得到的结果:
true
false
true
false
为什么会得到这样的一个结果呢?我们一步一步的分析。
第一、str5.equals(str3)这个结果为true,不用太多的解释,因为字符串的值的内容相同。
第二、str5 == str3对比的是引用的地址是否相同,由于str5采用new String方式定义的,所以地址引用一定不相等。所以结果为false。
第三、当str5调用intern的时候,会检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用。
第四,当str4 = str1 + str2后,str4的值也为”ab”,但是为什么这个结果会是false呢?先看下面代码:
1 String a = new String( " ab " );
2 String b = new String( " ab " );
3 String c = " ab " ;
4 String d = " a " + " b " ;
5 String e = " b " ;
6 String f = " a " + e;
7
8 System.out.println(b.intern() == a);
9 System.out.println(b.intern() == c);
10 System.out.println(b.intern() == d);
11 System.out.println(b.intern() == f);
12 System.out.println(b.intern() == a.intern());
运行结果:
false
true
true
false
true
由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。
当研究到这个地方的时候,突然想起来经常遇到的一个比较经典的Java问题,就是对比equal和==的区别,当时记得老师只是说“==”判断的是“地址”,但是并没说清楚什么时候会有地址相等的情况。现在看来,在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用。
执行下面的代码:
1 String a = " abc " ;
2 String b = " abc " ;
3 String c = " a " + " b " + " c " ;
4 String d = " a " + " bc " ;
5 String e = " ab " + " c " ;
6
7 System.out.println(a == b);
8 System.out.println(a == c);
9 System.out.println(a == d);
10 System.out.println(a == e);
11 System.out.println(c == d);
12 System.out.println(c == e);
运行的结果:
true
true
true
true
true
true