码迷,mamicode.com
首页 > Web开发 > 详细

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

时间:2014-11-03 12:49:28      阅读:128      评论:0      收藏:0      [点我收藏+]

标签:style   blog   http   color   ar   os   使用   for   sp   

 

概述

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

不可改变对象

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

 

 1bubuko.com,布布扣using System;
 2bubuko.com,布布扣
 3bubuko.com,布布扣namespace Demo1
 4bubuko.com,布布扣{
 5bubuko.com,布布扣    /// <summary>
 6bubuko.com,布布扣    /// String连接测试
 7bubuko.com,布布扣    /// </summary>

 8bubuko.com,布布扣    public class Test
 9bubuko.com,布布扣    {
10bubuko.com,布布扣        public static void Main(string[] args)
11bubuko.com,布布扣        {
12bubuko.com,布布扣            string a = "1234";
13bubuko.com,布布扣            Console.WriteLine(a);
14bubuko.com,布布扣
15bubuko.com,布布扣            a += "5678";
16bubuko.com,布布扣            Console.WriteLine(a);
17bubuko.com,布布扣            Console.ReadLine();
18bubuko.com,布布扣        }

19bubuko.com,布布扣    }

20bubuko.com,布布扣}

21bubuko.com,布布扣

 

运行的结果:

bubuko.com,布布扣1234
bubuko.com,布布扣
bubuko.com,布布扣
12345678
bubuko.com,布布扣

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

引用类型

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

 

 1bubuko.com,布布扣using System;
 2bubuko.com,布布扣
 3bubuko.com,布布扣namespace Demo2
 4bubuko.com,布布扣{
 5bubuko.com,布布扣    /// <summary>
 6bubuko.com,布布扣    /// String引用类型测试
 7bubuko.com,布布扣    /// </summary>

 8bubuko.com,布布扣    public class Test
 9bubuko.com,布布扣    {
10bubuko.com,布布扣        public static void Main(string[] args)
11bubuko.com,布布扣        {
12bubuko.com,布布扣            string a = "1234";
13bubuko.com,布布扣
14bubuko.com,布布扣            Console.WriteLine(a);
15bubuko.com,布布扣
16bubuko.com,布布扣            Test.Change(a);
17bubuko.com,布布扣
18bubuko.com,布布扣            Console.WriteLine(a);
19bubuko.com,布布扣            Console.ReadLine();
20bubuko.com,布布扣        }

21bubuko.com,布布扣
22bubuko.com,布布扣        public static void Change(string s)
23bubuko.com,布布扣        {
24bubuko.com,布布扣            s = "5678";
25bubuko.com,布布扣        }

26bubuko.com,布布扣    }

27bubuko.com,布布扣}

运行结果:

bubuko.com,布布扣1234
bubuko.com,布布扣
bubuko.com,布布扣
1234

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

 1bubuko.com,布布扣using System;
 2bubuko.com,布布扣
 3bubuko.com,布布扣namespace Demo2
 4bubuko.com,布布扣{
 5bubuko.com,布布扣    /// <summary>
 6bubuko.com,布布扣    /// String引用类型测试
 7bubuko.com,布布扣    /// </summary>

 8bubuko.com,布布扣    public class Test
 9bubuko.com,布布扣    {
10bubuko.com,布布扣        public static void Main(string[] args)
11bubuko.com,布布扣        {
12bubuko.com,布布扣            string a = "1234";
13bubuko.com,布布扣
14bubuko.com,布布扣            Console.WriteLine(a);
15bubuko.com,布布扣
16bubuko.com,布布扣            Test.Change(ref a);
17bubuko.com,布布扣
18bubuko.com,布布扣            Console.WriteLine(a);
19bubuko.com,布布扣            Console.ReadLine();
20bubuko.com,布布扣        }

21bubuko.com,布布扣
22bubuko.com,布布扣        public static void Change(ref string s)
23bubuko.com,布布扣        {
24bubuko.com,布布扣            s = "5678";
25bubuko.com,布布扣        }

26bubuko.com,布布扣    }

27bubuko.com,布布扣}

28bubuko.com,布布扣

运行结果:

bubuko.com,布布扣1234
bubuko.com,布布扣
bubuko.com,布布扣
5678

 

字符串的比较

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

 

 

 

 1bubuko.com,布布扣using System;
 2bubuko.com,布布扣
 3bubuko.com,布布扣namespace Demo3
 4bubuko.com,布布扣{
 5bubuko.com,布布扣    /// <summary>
 6bubuko.com,布布扣    /// String类型的比较
 7bubuko.com,布布扣    /// </summary>

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

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

37bubuko.com,布布扣            end = Environment.TickCount;
38bubuko.com,布布扣            Console.WriteLine((end-start));
39bubuko.com,布布扣
40bubuko.com,布布扣            Console.ReadLine();
41bubuko.com,布布扣        }

42bubuko.com,布布扣    }

43bubuko.com,布布扣}

44bubuko.com,布布扣

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

bubuko.com,布布扣1671
bubuko.com,布布扣
bubuko.com,布布扣
4172

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

1bubuko.com,布布扣public static bool operator ==(string a, string b)
2bubuko.com,布布扣{
3bubuko.com,布布扣      return string.Equals(a, b);
4bubuko.com,布布扣}

5bubuko.com,布布扣

 1bubuko.com,布布扣public static bool Equals(string a, string b)
 2bubuko.com,布布扣{
 3bubuko.com,布布扣      if (a == b)
 4bubuko.com,布布扣      {
 5bubuko.com,布布扣            return true;
 6bubuko.com,布布扣      }

 7bubuko.com,布布扣      if ((a != null&& (b != null))
 8bubuko.com,布布扣      {
 9bubuko.com,布布扣            return a.Equals(b);
10bubuko.com,布布扣      }

11bubuko.com,布布扣      return false;
12bubuko.com,布布扣}

13bubuko.com,布布扣

字符串驻留

看一下这段代码:

 

 1bubuko.com,布布扣using System;
 2bubuko.com,布布扣
 3bubuko.com,布布扣namespace Demo4
 4bubuko.com,布布扣{
 5bubuko.com,布布扣    /// <summary>
 6bubuko.com,布布扣    /// String的驻留
 7bubuko.com,布布扣    /// </summary>

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

23bubuko.com,布布扣    }

24bubuko.com,布布扣}

25bubuko.com,布布扣

 

执行的结果:

bubuko.com,布布扣False
bubuko.com,布布扣
bubuko.com,布布扣True

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

StringBuilder对象

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

 

 1bubuko.com,布布扣using System;
 2bubuko.com,布布扣using System.Text;
 3bubuko.com,布布扣
 4bubuko.com,布布扣namespace Demo5
 5bubuko.com,布布扣{
 6bubuko.com,布布扣    /// <summary>
 7bubuko.com,布布扣    /// String和StringBulider比较
 8bubuko.com,布布扣    /// </summary>

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

25bubuko.com,布布扣            end = Environment.TickCount;
26bubuko.com,布布扣            Console.WriteLine((end-start));
27bubuko.com,布布扣            
28bubuko.com,布布扣            ///测试StringBuilder所用的时间
29bubuko.com,布布扣            start = Environment.TickCount;
30bubuko.com,布布扣            for(int i=0;i<times;i++)
31bubuko.com,布布扣            {
32bubuko.com,布布扣                s.Append(i.ToString());
33bubuko.com,布布扣            }

34bubuko.com,布布扣            end = Environment.TickCount;
35bubuko.com,布布扣            Console.WriteLine((end-start));
36bubuko.com,布布扣
37bubuko.com,布布扣            Console.ReadLine();
38bubuko.com,布布扣        }

39bubuko.com,布布扣    }

40bubuko.com,布布扣}

41bubuko.com,布布扣

运行结果:

bubuko.com,布布扣884
bubuko.com,布布扣
bubuko.com,布布扣
0

 

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

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

标签:style   blog   http   color   ar   os   使用   for   sp   

原文地址:http://www.cnblogs.com/SamFlynn/p/4070847.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!