String s; String s = null; String s = ""; .intern()

String abc=null;String abc="";String abc;三种写法有什么区别?

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



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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值