基本

□ 哪些属于引用类型

类(object,string),接口、数组、委托

□ 引用类型分配在哪里

● 引用类型变量位于线程栈。
● 引用类型实例分配在托管堆上。
● 当引用类型实例的大小小于85000bytes,被分配在GC堆上,当大于或等于85000bytes,被分配在LOH(Large Object Heap)上。

□ 变量(Variable),对象(Object),实例(Instance)

变量:
变量分配在线程栈上。
变量可以是值类型,也可以是引用类型。
当变量是引用类型时,包含了对对象的引用(内存地址),也叫做"对象引用"。

对象:
对类、接口、委托和数组等的一个抽象描述。

实例:
在堆上创建的对象,称为对象实例。

□ 引用类型没有new意味着什么?

Object a = null;
Console.WriteLine(a.ToString());
运行报错"未将对象引用设置到对象实例"。
意思是,在线程栈上创建的变量a没有指向到堆上的对象实例。

□ 托管堆上的垃圾回收

GC会遍历所有托管堆上的对象,按照一定的递归遍历算法,对那些没有被引用的不可访问对象实施回收。

new的背后发生了什么

展开 class Program { static void Main(string[] args) { Person p; p = new Person(20); } }public class Person { public int _age;public Person(int age) { _age = age; }public Person() { } }

另外,引用类型的值,比如这里的引用类型Person中的值_age也被分配在托管堆上。

线程栈上的2个变量引用同一个对象实例的内存地址

线程栈上的2个变量引用同一个对象实例的内存地址,改变其中一个变量的值会影响到另外一个变量。

展开    class Program { static void Main(string[] args) { Person p1 = new Person(20); Person p2 = p1; Console.WriteLine("没有改变时p2的年龄是:" + p2._age + "岁"); p1._age = 18; Console.WriteLine("改变p1的值,p2的年龄也被改变了,现在是:" + p2._age + "岁,真好,年轻了!"); Console.ReadKey(); } }public class Person { public int _age;public Person(int age) { _age = age; }public Person() { } }

string类型是特殊的引用类型

□ 特殊性体现在

从应用角度体现了值类型语义,从内存角度实现为引用类型存储,位于托管堆。

□ 什么是string
可以看作是char的集合char[]

□ string创建与实例化

string str = "Hello";

以下错误
String str = new String("Hello");
因为System.String没有提供此构造函数

以下可以
Char[] cs = {'a', 'b','c'};
String strArr = new String(cs);
但很少使用这种方式。

□ 字符串的恒定性Immutability

是指字符串一经创建,就不可改变。
字符串一旦创建,在托管堆上分配一块连续的内存空间。

恒定性的好处:
对String对象的任意操作,不会改变原字符串。
操作字符串不会出现线程同步的问题。
成就了字符串驻留。

恒定性的不足:
因为恒定性,对字符串的任何操作,比如字符串比较,字符串链接,字符串格式化等都会创建新的字符串,这样造成内存与性能的双重损耗。如下:

public static void Main() 
{ 
    string str = "This is a test about immuntablility of string type."; 
    Console.WriteLine(str.Inseert(0,"Hi").Substring(19).ToUpper()); 
    Console.WriteLine(str); 
} 

由于Insert,Substring,ToUpper这些方法,都会创建出新的临时字符串,而这些新的字符串不被其他代码引用的时候,就会被垃圾回收,造成性能上的损失。

恒定性的前提,String为密封类:

public sealed class String:IComparable, ICloneable,IConvertible,IComparable<string>,IEnumerable<char>,IEnumerable,IEquatable<string>

□ 字符串驻留String Interning

MSDN对于字符驻留的定义:公共语言运行库通过维护一个哈希表(Hash Table)来存放字符串,该表成为拘留池,也叫驻留池。

字符串驻留弥补了恒定性的不足:
对于相同的字符串,CLR不会不会为其分配内存空间,而是共享同一内存。
CLR内部维护了一个哈希表HashTable来管理其创建的大部分String对象。key是string本身,value是string对应的内存地址。

驻留的2个静态方法:
public static string Intern(string str);
当str位于作为key位于CLR的驻留池时,返回对str的引用,否则将str字符串添加到hash table中,作为key,并返回引用。

public static string IsInterned(string str);
当str位于作为key位于CLR的驻留池时,返回对str的引用,否则返回null引用,也不添加到hash table中。

□ 字符串驻留是进程级的

可以跨应用程序域AppDomain而存在,驻留池在CLR加载时创建,分配在System Domain中,被进程所有AppDomain所共享,其生命周期不受GC控制。

例子1:

        static void Main(string[] args) 
        { 
            string strA = "ab"; 
            string strB = "ab"; 
            Console.WriteLine(ReferenceEquals(strA,strB)); 
            string strC = "a"; 
            string strD = strC + "b"; 
            Console.WriteLine(ReferenceEquals(strA, strD)); 
            strD = String.Intern(strD); 
            Console.WriteLine(ReferenceEquals(strA,strD)); 
            Console.ReadKey(); 
        }

返回:
true
false
true

分析:
● strA与strB内容相同,在hash table中的key相同,对应了一样的引用地址,所以返回true。
● strD的内容虽然与strA相同,但由于是动态生成的,不会把hash table中key为ab的引用地址赋值给strD,所以strA与strD引用地址不一样,返回false。
● strD = String.Intern(strD);手动对strD实施驻留,并发现hash table中已经有了ab这个key,就把对应的引用地址赋值给了strD,这样,strA与strD引用地址相同,返回true。

例子2:

        static void Main(string[] args) 
        { 
            string s1 = "abc"; 
            string s2 = "ab"; 
            string s3 = s2 + "c"; 
            Console.WriteLine(string.IsInterned(s3) ?? "null"); 
            Console.WriteLine(ReferenceEquals(s1,s3)); 
            Console.ReadKey(); 
        }

返回:
abc
false

分析:
● string.IsInterned(s3)对s3进行手动驻留,发现hash table中abc这个key,于是,就返回abc的引用地址。但并没有把引用地址赋值给s3。
● s1和s3的引用地址还是不一样,返回false。

例子3:

        static void Main(string[] args) 
        { 
            string strA = "abcdef"; 
            string strC = "abc"; 
            string strD = strC + "def"; 
            Console.WriteLine(ReferenceEquals(strA,strD)); 
            string StrE = "abc" + "def"; 
            Console.WriteLine(ReferenceEquals(strA,StrE)); 
            Console.ReadKey(); 
        }

返回:
False,       
true

分析:
● 因为strD = strC + "def"中strD的内容虽然与strA想同,但因为是动态生成的,不会被添加到hash table中,所以引用地址不一样,返回false。
●  "abc"+"def在IL中呈现为abcdef,不是动态生成的,并且发现hash table中已经有了abcdef这个key,于是就把对应的引用地址赋值给了strE,这样strA和StrE就有了相同的引用地址,返回true。如图:

例子4:

        static void Main(string[] args) 
        { 
            string s = "abc"; 
            //IsInterned()获取字符串变量的引用 
            Console.WriteLine(string.IsInterned(s) ?? "null"); 
            Console.ReadKey(); 
        }

返回"abc"。
分析:通过string s = "abc"使得hash table中有abc这个key,当进行string.IsInterned(s)手动驻留判断的时候,发现有abc这个key,就把对应的引用地址返回。

例子5:

    class Program 
    { 
        static void Main(string[] args) 
        { 
            string s = "ab"; 
            s += "c"; 
            //IsInterned()获取字符串变量的引用 
            Console.WriteLine(string.IsInterned(s) ?? "null"); 
            Console.ReadKey(); 
        } 
    }

返回null
分析:通过string s = "ab";使得hash table中有了ab这个key,s += "c"是动态拼接,不会把abc放到hashtable中,所以当通过string.IsInterned(s)手动驻留判断的时候,发现没有abc这个key,就返回null。

参考资料
※ 《.NET之美》--张子阳,感谢写了这么好的书!
※ 《你必须知道的.NET》--王涛,worktile创始人,感谢!

转载于:https://www.cnblogs.com/darrenji/p/3599550.html

05引用类型以及特殊引用类型string相关推荐

  1. 内置对象的API Array数组对象 String字符串对象 json字符串 JSON对象 js作用域及变量预解析 引用类型与值类型区别 共享引用 基本包装类型 数组去重

    01-内置对象的API a.Date对象获取时间 b.Array对象数组加工 c.String对象字符串加工 d.json字符串的语法格式 e.JSON对象的字符串与对象转换应用 02-JS作用域 a ...

  2. swift string转int_swift中结构体和类的区别(值类型和引用类型的区别)

    在swift中结构体和类有着更多的相同之处,在一般的使用中能够做到互相替换.我们可以先看看官方文档的描述: Unlike other programming languages, Swift does ...

  3. String 是值类型还是引用类型

    System.String继承自System.Object的一个子类,所有直接或简介继承自System .ValueType的类型属于值类型,这是判断值类型与引用类型的标准. string属于引用类型 ...

  4. 变量/值类型/引用类型/常量/枚举

    变量 声明语法 datatype identifier; 如:int i; //声明一个int类型的变量,但是在没有初始化之前编译器不允许使用该变量 同时声明多个 int a,b;//同时声明两个in ...

  5. JavaScript高级程序设计学习笔记--引用类型

    Object类型 对象字面量表示法: var person={ name:"Nicholas", age:29, 5:true }; 这人例子会创建一个对象,包含三个属性:name ...

  6. 转:图解C#的值类型,引用类型,栈,堆,ref,out

    C# 的类型系统可分为两种类型,一是值类型,一是引用类型,这个每个C#程序员都了解.还有托管堆,栈,ref,out等等概念也是每个C#程序员都会接触到的概念,也是C#程序员面试经常考到的知识,随便搜搜 ...

  7. C#中的值类型和引用类型

    文章目录 1 C#中的值类型 1.1 值类型示例程序 1.2 值类型(基本数据类型)的变量使用特点 2 C#中的引用类型 2.1 引用类型数据程序示例 2.2 引用数据类型的变量使用特点 3 变量类型 ...

  8. C# 值类型与引用类型的详解

    值类型与引用类型分这几种情况: 1.内存分为堆和栈,值类型的数据存储在栈中,引用类型的数据存储在堆中. 2.int numb=10,代码中的10是值类型的数据,numb只是一个指向10的变量而已.其中 ...

  9. ext如何将值存入变量_变量类型之值类型与引用类型

    前言 变量类型在我们日常开发中经常接触到,但是js中的变量类型与其他强类型语言不同,由于js是弱类型语言,因此他的变量拷贝在我们实际的日常开发中有很多需要注意的项.而半斤在最近的开发中遇到了很多匪夷所 ...

最新文章

  1. make life colorful
  2. queued frame 造成图形性能卡顿
  3. 图解CodeSmith使用和实用教程一 - 入门和生成MIS项目实体层代码
  4. java编译器分析_Java反编译器的剖析
  5. Leetcode--198. 打家劫舍
  6. 计组之数据运算:6、原码乘法运算
  7. 如何在cocoapods中使用更新版本的pod
  8. 开源 CMS系统 / SNS系统 / BBS系统
  9. 《WiscKey: Separating Keys from Values in SSD-conscious Storage》阅读笔记
  10. react-native 创建新的项目
  11. 抽奖活动的奖品怎么设置?
  12. 初探iOS项目使用MVP模式
  13. 市场调研-全球与中国LED多层指示灯市场现状及未来发展趋势
  14. 牛客14709 奇♂妙拆分 枚举
  15. Html读取本地文件夹下图片并显示的示例代码
  16. skipping incompatible xxxx.a when searching for -lxxx问题的解决
  17. 推进线上线下深度融合,5G+VR+直播技术助力文旅产业加快复苏
  18. Python和Pycharm运行环境有关问题详解
  19. 路由器上的usb接口有什么用_解决USB接口不够用,毕亚兹Type-C扩展坞体验测评
  20. Windows 挂载nfs

热门文章

  1. JavaScript学习(二十六)—事件处理程序的添加与删除
  2. html页面获取时间格式,js实现动态获取系统时间,显示到页面上
  3. Hay Points
  4. 最惊艳你的短句是什么?
  5. 怎样才能减少汽车油耗呢?
  6. 红豆、绿豆、黑豆、花生、莲子、薏仁米放在一起吃,可以吗?
  7. 手机开启热点给其他设备上网和用插卡随身路由给其他设备上网有何区别呢?
  8. 要想完全放弃Windows使用Linux需要多少勇气?
  9. 戴AirPods pro走路会传来咚咚的脚步声,有什么办法能消除这个声音?
  10. Qt——P7 对象树