日期:2013-02-03  浏览次数:20518 次

目的

本文的目的在于揭示和DOTNET及C#相关的一些常见的和不常见的问题。在这些问题中我的第一篇文章和string数据类型有关,string数据类型是一种引用类型,但是当和其他引用类型比较的时候,很多开发人员可能并不能完全理解它的行为。

问题

对于常见的引用类型,当改变一个对象别名的值时,这种变化也同样会在一个实际的对象中表现出来;反之亦然。但是对于string类型,似乎不是这样的。

解释

引用类型

假设我们有一个类MyType,这个类有一个属性Name;我们还有一个类AppType,这个类提供Main()方法来运行这个程序。

下面,我们来看看代码:



using System;

class MyType

{

private string name;

public string Name

{

set

{

name=value;

}

get

{

return name;

}

}

}

class AppType

{

public static void Main()

{

MyType obj1,obj2;

Console.WriteLine("*****Learning reference Philosophy*****");

obj2=new MyType();

obj2.Name="Sadiq";

obj1=obj2;

Console.WriteLine("values of obj1={0} and obj2={1}",obj1.Name,obj2.Name);

obj1.Name="Ahmed";

Console.WriteLine("values of obj1={0} and obj2={1}",obj1.Name,obj2.Name);

}

}


当你编译并且运行这段代码时,你将得到如下输出:

*****Learning reference philosophy*****
values of obj1=Sadiq and obj2=Sadiq
values of obj1=Ahmed and obj2=Ahmed

这表明obj1不过是obj2的别名,换句话说,obj1和obj2都指向同一个内存空间。

值类型

和上面的代码差不多,不同的是这次我们将MyType定义为类,其他部分都相同,我们先看看代码:

using System;

struct MyType

{

private string name;

public string Name

{

set

{

name=value;

}

get

{

return name;

}

}

}

class AppType

{

public static void Main()

{

MyType obj1,obj2;

Console.WriteLine("*****Learning reference Philosophy*****");

obj2=new MyType();

obj2.Name="Sadiq";

obj1=obj2;

Console.WriteLine("values of obj1={0} and obj2={1}",obj1.Name,obj2.Name);

obj1.Name="Ahmed";

Console.WriteLine("values of obj1={0} and obj2={1}",obj1.Name,obj2.Name);

}

}


我们再来看看上面代码运行后的输出:

*****Learning reference philosophy*****
values of obj1=Sadiq and obj2=Sadiq
values of obj1=Ahmed and obj2=Sadiq

这表明obj1和obj2并不相同,也就是说,他们指向不同的内存空间。

引用类型还是值类型?
现在,让我们看看直接使用string类型的情况:

using System;

class AppType

{

public static void Main()

{

String obj1,obj2;

Console.WriteLine("*****Learning reference philosophy*****");

//No need of it

//obj2=new MyType();

obj2="Sadiq";

obj1=obj2;

Console. WriteLine("values of obj1={0} and obj2={1}",obj1,obj2);

obj1="Ahmed";

Console.WriteLine("values of obj1={0} and obj2={1}",obj1,obj2);

}

}


当你运行这段代码,你会得到:

*****Learning reference philosophy*****
values of obj1=Sadiq and obj2=Sadiq
values of obj1=Ahmed and obj2=Sadiq

这表明obj1并不是obj2的别名,即obj1和obj2指向不同的内存空间。

非常奇怪!确实!我们都知道string类型是动态增长的,这表明它必须在堆上分配内存。我们都知道引用类型都在堆上分配内存,那么string类型也应该是引用类型,那么为何它又表现出和值类型一样的性质呢?

原因
关键在于如下的两行代码中:

string obj1;
obj1 = “value forces to allocate a memory”;

第一行代码仅仅是定义了一个对象