你真的了解.NET中的String吗?

 

你真的了解.NET中的String吗?<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

Terrylee20051225

概述

String在任何语言中,都有它的特殊性,在.NET中也是如此。它属于基本数据类型,也是基本数据类型中唯一的引用类型。字符串可以声明为常量,但是它却放在了堆中。希望通过本文能够使大家对.NET中的String有一个深入的了解。

不可改变对象

.NETString是不可改变对象,一旦创建了一个String对象并为它赋值,它就不可能再改变,也就是你不可能改变一个字符串的值。这句话初听起来似乎有些不可思议,大家也许马上会想到字符串的连接操作,我们不也可以改变字符串吗?看下面这段代码:

 1 None.gif using  System;
 2 None.gif
 3 None.gif namespace  Demo1
 4 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 5ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 6InBlock.gif    /// String连接测试
 7ExpandedSubBlockEnd.gif    /// </summary>

 8InBlock.gif    public class Test
 9ExpandedSubBlockStart.gifContractedSubBlock.gif    dot.gif{
10InBlock.gif        public static void Main(string[] args)
11ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
12InBlock.gif            string a = "1234";
13InBlock.gif            Console.WriteLine(a);
14InBlock.gif
15InBlock.gif            a += "5678";
16InBlock.gif            Console.WriteLine(a);
17InBlock.gif            Console.ReadLine();
18ExpandedSubBlockEnd.gif        }

19ExpandedSubBlockEnd.gif    }

20ExpandedBlockEnd.gif}

21 None.gif

运行的结果:

None.gif 1234
None.gif
None.gif
12345678
None.gif

看起来我们似乎已经把MyStr的值从“1234”改为了“12345678”。事实是这样的吗?实际上并没有改变。在第5行代码中创建了一个String对象它的值是“1234”,MyStr指向了它在内存中的地址;第七行代码中创建了一个新的String对象它的值是“12345678”,MyStr指向了新的内存地址。这时在堆中其实存在着两个字符串对象,尽管我们只引用了它们中的一个,但是字符串“1234”仍然在内存中驻留。

引用类型

前面说过String是引用类型,这就是如果我们创建很多个相同值的字符串对象,它在内存中的指向地址应该是一样的。也就是说,当我们创建了字符串对象a,它的值是“1234”,当我们再创建一个值为“1234”的字符串对象b时它不会再去分配一块内存空间,而是直接指向了a在内存中的地址。这样可以确保内存的有效利用。看下面的代码:

 1 None.gif using  System;
 2 None.gif
 3 None.gif namespace  Demo2
 4 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 5ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 6InBlock.gif    /// String引用类型测试
 7ExpandedSubBlockEnd.gif    /// </summary>

 8InBlock.gif    public class Test
 9ExpandedSubBlockStart.gifContractedSubBlock.gif    dot.gif{
10InBlock.gif        public static void Main(string[] args)
11ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
12InBlock.gif            string a = "1234";
13InBlock.gif
14InBlock.gif            Console.WriteLine(a);
15InBlock.gif
16InBlock.gif            Test.Change(a);
17InBlock.gif
18InBlock.gif            Console.WriteLine(a);
19InBlock.gif            Console.ReadLine();
20ExpandedSubBlockEnd.gif        }

21InBlock.gif
22InBlock.gif        public static void Change(string s)
23ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
24InBlock.gif            s = "5678";
25ExpandedSubBlockEnd.gif        }

26ExpandedSubBlockEnd.gif    }

27ExpandedBlockEnd.gif}

运行结果:

None.gif 1234
None.gif
None.gif
1234

做一个小改动,注意Change(ref string s)

 1 None.gif using  System;
 2 None.gif
 3 None.gif namespace  Demo2
 4 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 5ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 6InBlock.gif    /// String引用类型测试
 7ExpandedSubBlockEnd.gif    /// </summary>
 8InBlock.gif    public class Test
 9ExpandedSubBlockStart.gifContractedSubBlock.gif    dot.gif{
10InBlock.gif        public static void Main(string[] args)
11ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
12InBlock.gif            string a = "1234";
13InBlock.gif
14InBlock.gif            Console.WriteLine(a);
15InBlock.gif
16InBlock.gif            Test.Change(ref a);
17InBlock.gif
18InBlock.gif            Console.WriteLine(a);
19InBlock.gif            Console.ReadLine();
20ExpandedSubBlockEnd.gif        }

21InBlock.gif
22InBlock.gif        public static void Change(ref string s)
23ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
24InBlock.gif            s = "5678";
25ExpandedSubBlockEnd.gif        }

26ExpandedSubBlockEnd.gif    }

27ExpandedBlockEnd.gif}

28 None.gif

运行结果:

None.gif 1234
None.gif
None.gif
5678

字符串的比较

.NET中,对字符串的比较操作并不仅仅是简单的比较二者的值,= =操作首先比较两个字符串的引用,如果引用相同,就直接返回True;如果不同再去比较它们的值。所以如果两个值相同的字符串的比较相对于引用相同的字符串的比较要慢,中间多了一步判断引用是否相同。看下面这段代码:

 1 None.gif using  System;
 2 None.gif
 3 None.gif namespace  Demo3
 4 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 5ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 6InBlock.gif    /// String类型的比较
 7ExpandedSubBlockEnd.gif    /// </summary>

 8InBlock.gif    public class Test
 9ExpandedSubBlockStart.gifContractedSubBlock.gif    dot.gif{
10InBlock.gif        public static void Main(string[] args)
11ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
12InBlock.gif            string a = "1234";
13InBlock.gif            string b = "1234";
14InBlock.gif            string c = "123";
15InBlock.gif            c += "4";
16InBlock.gif
17InBlock.gif            int times = 1000000000;
18InBlock.gif            int start,end;
19InBlock.gif            
20ExpandedSubBlockStart.gifContractedSubBlock.gif            /**////测试引用相同所用的实际时间
21InBlock.gif            start = Environment.TickCount;
22InBlock.gif            for(int i=0;i<times;i++)
23ExpandedSubBlockStart.gifContractedSubBlock.gif            dot.gif{
24InBlock.gif                if(a==b)
25ExpandedSubBlockStart.gifContractedSubBlock.gif                dot.gif{}
26ExpandedSubBlockEnd.gif            }

27InBlock.gif            end = Environment.TickCount;
28InBlock.gif            Console.WriteLine((end-start));
29InBlock.gif            
30ExpandedSubBlockStart.gifContractedSubBlock.gif            /**////测试引用不同而值相同所用的实际时间
31InBlock.gif            start = Environment.TickCount;
32InBlock.gif            for(int i=0;i<times;i++)
33ExpandedSubBlockStart.gifContractedSubBlock.gif            dot.gif{
34InBlock.gif                if(a==c)
35ExpandedSubBlockStart.gifContractedSubBlock.gif                dot.gif{}
36ExpandedSubBlockEnd.gif            }

37InBlock.gif            end = Environment.TickCount;
38InBlock.gif            Console.WriteLine((end-start));
39InBlock.gif
40InBlock.gif            Console.ReadLine();
41ExpandedSubBlockEnd.gif        }

42ExpandedSubBlockEnd.gif    }

43ExpandedBlockEnd.gif}

44 None.gif

执行的结果(运行的结果可能有些不同):

None.gif 1671
None.gif
None.gif
4172

由此我们看出值相同时的比较用= =比引用相同时的比较慢了好多。这里仅仅是一个测试,因为做这样的比较并没有任何实际的意义。
有一点需要明确的是,.NET中==跟Equals()内部机制完全是一样的,==是它的一个重载。

1 None.gif public   static   bool   operator   == ( string  a,  string  b)
2 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
3InBlock.gif      return string.Equals(a, b);
4ExpandedBlockEnd.gif}

5 None.gif

 1 None.gif public   static   bool  Equals( string  a,  string  b)
 2 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 3InBlock.gif      if (a == b)
 4ExpandedSubBlockStart.gifContractedSubBlock.gif      dot.gif{
 5InBlock.gif            return true;
 6ExpandedSubBlockEnd.gif      }

 7InBlock.gif      if ((a != null&& (b != null))
 8ExpandedSubBlockStart.gifContractedSubBlock.gif      dot.gif{
 9InBlock.gif            return a.Equals(b);
10ExpandedSubBlockEnd.gif      }

11InBlock.gif      return false;
12ExpandedBlockEnd.gif}

13 None.gif

字符串驻留

看一下这段代码:

 1 None.gif using  System;
 2 None.gif
 3 None.gif namespace  Demo4
 4 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 5ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 6InBlock.gif    /// String的驻留
 7ExpandedSubBlockEnd.gif    /// </summary>

 8InBlock.gif    public class Test
 9ExpandedSubBlockStart.gifContractedSubBlock.gif    dot.gif{
10InBlock.gif        public static void Main(string[] args)
11ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
12InBlock.gif            string a = "1234";
13InBlock.gif            string s = "123";
14InBlock.gif            s += "4";
15InBlock.gif
16InBlock.gif            string b = s;
17InBlock.gif            string c = String.Intern(s);
18InBlock.gif
19InBlock.gif            Console.WriteLine((object)a == (object)b);
20InBlock.gif            Console.WriteLine((object)a == (object)c);
21InBlock.gif            Console.ReadLine();
22ExpandedSubBlockEnd.gif        }

23ExpandedSubBlockEnd.gif    }

24ExpandedBlockEnd.gif}

25 None.gif

执行的结果:

None.gif False
None.gif
None.gifTrue

在这段代码中,比较这两个对象发现它的引用并不是一样的。如果要想是它们的引用相同,可以用Intern()函数来进行字符串的驻留(如果有这样的值存在)。

StringBuilder对象

通过上面的分析可以看出,String类型在做字符串的连接操作时,效率是相当低的,并且由于每做一个连接操作,都会在内存中创建一个新的对象,占用了大量的内存空间。这样就引出StringBuilder对象,StringBuilder对象在做字符串连接操作时是在原来的字符串上进行修改,改善了性能。这一点我们平时使用中也许都知道,连接操作频繁的时候,使用StringBuilder对象。但是这两者之间的差别到底有多大呢?来做一个测试:

 1 None.gif using  System;
 2 None.gif using  System.Text;
 3 None.gif
 4 None.gif namespace  Demo5
 5 ExpandedBlockStart.gifContractedBlock.gif dot.gif {
 6ExpandedSubBlockStart.gifContractedSubBlock.gif    /**//// <summary>
 7InBlock.gif    /// String和StringBulider比较
 8ExpandedSubBlockEnd.gif    /// </summary>

 9InBlock.gif    public class Test
10ExpandedSubBlockStart.gifContractedSubBlock.gif    dot.gif{
11InBlock.gif        public static void Main(string[] args)
12ExpandedSubBlockStart.gifContractedSubBlock.gif        dot.gif{
13InBlock.gif            string a = "";
14InBlock.gif            StringBuilder s = new StringBuilder();
15InBlock.gif
16InBlock.gif            int times = 10000;
17InBlock.gif            int start,end;
18InBlock.gif            
19ExpandedSubBlockStart.gifContractedSubBlock.gif            /**////测试String所用的时间
20InBlock.gif            start = Environment.TickCount;
21InBlock.gif            for(int i=0;i<times;i++)
22ExpandedSubBlockStart.gifContractedSubBlock.gif            dot.gif{
23InBlock.gif                a += i.ToString();
24ExpandedSubBlockEnd.gif            }

25InBlock.gif            end = Environment.TickCount;
26InBlock.gif            Console.WriteLine((end-start));
27InBlock.gif            
28ExpandedSubBlockStart.gifContractedSubBlock.gif            /**////测试StringBuilder所用的时间
29InBlock.gif            start = Environment.TickCount;
30InBlock.gif            for(int i=0;i<times;i++)
31ExpandedSubBlockStart.gifContractedSubBlock.gif            dot.gif{
32InBlock.gif                s.Append(i.ToString());
33ExpandedSubBlockEnd.gif            }

34InBlock.gif            end = Environment.TickCount;
35InBlock.gif            Console.WriteLine((end-start));
36InBlock.gif
37InBlock.gif            Console.ReadLine();
38ExpandedSubBlockEnd.gif        }

39ExpandedSubBlockEnd.gif    }

40ExpandedBlockEnd.gif}

41 None.gif

运行结果:

None.gif 884
None.gif
None.gif
0

通过上面的分析,可以看出用String来做字符串的连接时效率非常低,但并不是所任何情况下都要用StringBuilder,当我们连接很少的字符串时可以用String,但当做大量的或频繁的字符串连接操作时,就一定要用StringBuilder

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值