现在的位置: 首页 > 综合 > 正文

C# string 拥有值类型特点的特殊引用类型 “字符串具有恒等性”

2012年02月22日 ⁄ 综合 ⁄ 共 1967字 ⁄ 字号 评论关闭

String类型直接继承自Object,所以它是一个要new出来的引用类型,即线程的堆栈上不会驻留有任何字符串。

(所有的值类型都继承自System.ValueType。值得指出的是System.ValueType却是一个
引用类型)

代码一:
string str1 = "string"; string str2 = "stri"+"ng"; Console.WriteLine(string.ReferenceEquals(str1, str2));

既然String类型是引用类型,那么代码一输出的应该是False,然而事实上代码一输出时的是True。

这是因为当CLR初始化时,它会创建
一个内部的散列表
Key为字符串,Value为指向托管堆中字符串对象的引用

但是,动态创建的字符串不会去查询散列表,而是直接在托管堆中创建新的String对象。string str3 = "system." + str1;

当构造str1时,先会去散列表中查询是否存在”string”字符串,
如果不存在那么会在托管堆(heap)中构造一个新的String对象,然后将”string”字符串和指向该对象的引用添加到散列表中,当构造str2时,由于散列表中存在Key为”string”的引用,于是将Value值赋值给str2,那么str1和str2引用的是同一个String对象,代码一自然就返回
True了

代码二:
static void Main(string[] args) { string str = "string"; Change(str); Console.WriteLine(str); }

static void Change(string str) { str = "Changed"; }

方法传递的参数是原内容的拷贝,其过程如图所示:

语句str=”Changed”之前

diyblPic

语句str=”Changed”之后

diyblPic

这样可以看到原来String对象并未改变,str=”Changed”只是创建一个新的String对象(其它引用类型是改变内存地址1指向的值),因此这个方法的参数需要加上ref或者out修饰符

这里也可以得出字符串具有恒等性,也就是说一个字符串一旦被创建,我们就不能再将其变长、
变短、或者改变其中的任何字符。

代码三:
string str1 = "string"; string str2 = "system." + "string"; string str3 = "system." + str1; string str4 = "system.string";

Console.WriteLine(string.Equals(str2, str3)); True

Console.WriteLine(string.ReferenceEquals(str2, str3));  False

Console.WriteLine(string.Equals(str4, str2)); True

Console.WriteLine(string.ReferenceEquals(str2, str4)); True

根据代码一和二的分析,代码三的输出结果为:True True True True,然而事实却不是这样,正确的结果为:True False
True True。这是因为动态创建的字符串不会去查询散列表,而是直接在托管堆中创建新的String对象,如语句string str3 =
“syetem.”+str1
,因此用string.ReferenceEquals来比较str2和str3会返回False,而用
string.ReferenceEquals来比较str2和str4会返回True。当然可以将str3字符串手动加入到散列表中,并返回引
用:str3 =
string.Intern(str3),这样用string.ReferenceEquals来比较str2和str3会返回True,至于
string.Equals都返回True的原因是String重写了Equals方法,内部会先检查两个引用是否指向同一个对象,如果是返回True,
不是则再比较各个字符。

引用:http://www.diybl.com/course/4_webprogram/asp.net/netjs/20091107/181566.html

@, 克服了转义字符串带来的阅读障碍。

Double.Parse, TryParse(str, out num);  Convert.ToDouble, 异常处理方式不同。

按引用传递,ref参数传递之前必须初始化,out没要求。

sizeof获取值类型大小,不适用于引用类型。 值类型继承自ValueType, which inherits Object, but it overrides Equals方法,按值比较,instead of 引用地址。String也重载了Equals方法,比较值大小。

抱歉!评论已关闭.