目的
本文的目的在于揭示和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”;
第一行代码仅仅是定义了一个对象