Java String 实例研究


public class StringTest {

    /**
     * @param args
     */
    public static void main(String[] args) {
        /*
         *  做个基础的说明,堆(heap)内存和栈(Stack)内存的问题。
         *  堆和栈的数据结构这里就不解释了。
         *  Java语言使用内存的时候,栈内存主要保存以下内容:
         *  基本数据类型和对象的引用,而堆内存存储对象,栈内存的速度要快于堆内存。
         *  总结成一句话就是:引用在栈而对象在堆。
         *  Java中的比较有两种,是==和equals()方法,equals()是Object类的方法.
         *  
         *  String类重写了equals()方法,改变了这些类型对象相等的原则,即判断对象是否相等依据的原则为判断二者的内容是否相等。
         *  了解以上内容后我们来说说String,String类的本质是字符数组char[],其次String类是final的,是不可被继承的,
         *  这点可能被大多数人忽略,再次String是特殊的封装类型,使用String时可以直接赋值,也可以用new来创建对象,
         *  但是这二者的实现机制是不同的。还有一个String池的概念,Java运行时维护一个String池,池中的String对象不可重复,
         *  没有创建,有则作罢。String池不属于堆和栈,而是属于常量池。
         */
        String str = "abc";   
        String str1 = "abc";
        /*
         *  第一句的真正含义是在String池中创建一个对象”abc”,然后引用时str指向池中的对象”abc”。
         *  第二句执行时,因为”abc”已经存在于String池了,所以不再创建,则str==str1返回true就明白了。
         *  str1==”abc”肯定正确了,在String池中只有一个”abc”,而str和str1都指向池中的”abc”,就是这个道理。
         */
        String str2 = new String("abc");
        /*
         * 这个是Java SE的热点问题,众所周知,单独这句话创建了2个String对象,而基于上面两句,只在栈内存创建str2引用,
         * 在堆内存上创建一个String对象,内容是”abc”,而str2指向堆内存对象的首地址。      
         */
        System.out.println(str == str1);   
        System.out.println(str1 == "abc");   
        System.out.println(str2 == "abc");
        /*
         * 下面就是str2==”abc”的问题了,显然不对,”abc”是位于String池中的对象,而str2指向的是堆内存的String对象,
         * ==判断的是地址,肯定不等了。
         */
        System.out.println(str1 == str2);   
        System.out.println(str1.equals(str2));
        //str1.equals(str2),这个是对的,前面说过,String类的equals重写了Object类的equals()方法,实际就是判断内容是否相同了。
        System.out.println(str1 == str2.intern());   
        System.out.println(str2 == str2.intern());
        /*
         *  下面说下intern()方法,在JavaDoc文档中,这样描述了intern()方法:返回字符串对象的规范化表示形式。
         *  怎么理解这句话?实际上过程是这样进行的:该方法现在String池中查找是否存在一个对象,存在了就返回String池中对象的引用。
         *  那么本例中String池存在”abc”,则调用intern()方法时返回的是池中”abc”对象引用,那么和str/str1都是等同的,和str2就不同了,因为str2指向的是堆内存。
         */
        System.out.println(str1.hashCode() == str2.hashCode());
        //hashCode()方法是返回字符串内容的哈希码,既然内容相同,哈希码必然相同,那他们就相等了,这个容易理解。
        System.out.println(makinStrings());
        System.out.println(makinStrings1());
        stringappend();
    }
    
    private static String s_str = "abc";   

    public static void stringappend(){
        /*
         *  这个例子用来说明用+连接字符串时,实际上是在堆内容创建对象,那么combo指向的是堆内存存储”abc”字符串的空间首地址,
         *  显然str==combo是错误的,而str==combo.intern()是正确的,在String池中也存在”abc”,那就直接返回了,
         *  而str也是指向String池中的”abc”对象的。此例说明任何重新修改String都是重新分配内存空间,
         *  这就使得String对象之间互不干扰。也就是String中的内容一旦生成不可改变,直至生成新的对象。
         */
        String str1 = "a";   
        String str2 = "bc";   
        String combo = str1 + str2;   
        System.out.println(s_str == combo);   
        System.out.println(s_str == combo.intern());
        /*
         *  同时问题也来了,使用+连接字符串每次都生成新的对象,而且是在堆内存上进行,而堆内存速度比较慢(相对而言),
         *  那么再大量连接字符串时直接+是不可取的,当然需要一种效率高的方法。
         *  Java提供的StringBuffer和StringBuilder就是解决这个问题的。区别是前者是线程安全的而后者是非线程安全的,
         *  StringBuilder在JDK1.5之后才有。不保证安全的StringBuilder有比StringBuffer更高的效率。
         *  自JDK1.5之后,Java虚拟机执行字符串的+操作时,内部实现也是StringBuilder,之前采用StringBuffer实现。
         */
    }
    
    
    public static String makinStrings()
    {
        String s = "Fred"; // 创建一个String对象
        s = s + "47"; //编译器转成(new StringBuilder(String.valueOf(s))).append("47").toString();
        s = s.substring(2, 5);
        /*
         * 结论:创建了一个String对象,也许有很多人一开始就认为这条语句是创建了一个String对象,那么恭喜你,
         * 这条语句确实创建了一个String对象, 实际上就是substring方法创建了一个String对象。
         * 这也没什么复杂的,自已下一个JDK源代码,看看substring是如何实现的,就可以知道了。
         * 我先说一下吧。先不用管substring是如何实现的,反正在substring方法返回时使用了一个new显式地建立了一个String对象,
         * 不信自己看看源码。
         */
        s = s.toUpperCase(); //创建了一个String对象, toUpperCase()和substring方法类似,在返回时也是使用了new建立了一个String对象。
        return s.toString();//未创建String对象,String类的toString方法返回的就是this,因此,它的返回值就是s。
    }

    public static String makinStrings1()
    {
        /*
         * 弄清楚substring、toLowerCase和toUpperCase什么时候创建String对象,什么时候不创建对象。
         * substring方法在截取的子字符串长度等于原字符串时,直接返回原字符串。并不创建新的String对象。
         * toLowerCase方法在字符串中更本没有需要转换的大写字母时直接返回原字符串,如"abcd".toLowerCase()直接返回abcd,并不创建新的String对象
         * toUpperCase方法和toLowerCase类似。"ABCD".toUpperCase()直接返回ABCD。
         */
        String s = "Fred";     // 创建一个String对象
        s = s + "Iloveyou.".substring(1).toLowerCase();  // substring(1)创建一个String对象,由于toLowerCase()转换的字符串是"loveyou.",没有大写字母,因此,它不创建新的String对象
        s = s.substring(0);   // 由于substring(0)截获的是s本身,因此,这条语句不创建新的String对象
        s = s.substring(0,1).toUpperCase();  // substring(0,1)创建了一个String对象,但由于substring(0,1)的结果是"F",为一个大写字母,因此,toUpperCase直接返回"F"本身。
        return s.toString();
    }



}


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值