【读书笔记】C#高级编程 第十四章 内存管理和指针

(一)后台内存管理

1、值数据类型

Windows使用一个虚拟寻址系统,该系统把程序可用的内存地址映射到硬件内存中的实际地址,该任务由Windows在后台管理(32位每个进程可使用4GB虚拟内存,64位更多,这个内存包括可执行代码和加载的DLL,以及程序运行时使用的变量内容)。

在处理器的虚拟内存中,有一个区域称为栈。栈存储不是对象成员的值数据类型。

释放变量时,其顺序总是与它们分配内存的顺序相反,这就是栈的工作方式。

程序第一次运行时,栈指针指向为栈保留的内存块末尾。栈实际上是向下填充的,即从高内存地址向低内存地址填充。当数据入栈后,栈指针就会随之调整,以始终指向下一个空闲单元。

2、引用数据类型

托管堆使用一个方法(new运算符)来分配内存,再方法退出后很长一段时间存储其中的数据仍可用。与栈不同,堆上的内存是向上分配的。

建立引用变量的过程要比建立值类型的过程更复杂,其不能避免性能的系统开销。当一个引用变量超出作用域时,它会从栈中删除,但引用的数据仍保留在堆中,一直到程序终止,会垃圾回收器删除它为止,而只有在改数据不再被任何变量引用时,它才会被删除。

3、垃圾回收器

垃圾回收器释放了能释放的所有对象,就会把其他对象移动回堆的端部,再次形成一个连续的块。

堆的第一部分称为第0代,创建的新对象会移动到这个部分。垃圾回收器每运行一次后保留的对象被压缩后移动到下一代存放部分。

在.NET下,较大对象有自己的堆,称为大象堆。使用大于85000个字节的对象时,它们就会放到这个特殊的堆上。

第二代和大象堆上的回收现在放在后台线程上进行。

GCSettings.LatencyMode属性可以控制垃圾回收器进行垃圾回收的方式。

(二)释放非托管资源

垃圾回收器不知道如何释放非托管资源(文件句柄、网络连接、数据库连接),需要制定专门的规则,确保非托管资源在回收类的一个实例时释放。

  • 声明一个析构函数(或终结器),作为类的成员
  • 实现IDisposable接口

1、析构函数

析构函数的语法,没有返回类型,不带参数,没有访问修饰符,与类同名前面有一个波形符(~)。

class MyClass
{
    ~MyClass(){
        //析构函数
    }
}

C#析构函数无法确定何时执行。C#析构函数的实现会延迟对象最终从内存中删除的时间。

2、IDisposable接口

C#中,推荐使用System.IDisposable接口替代析构函数。IDisposable接口声明了一个Disposable()方法,它不带参数,返回void。

class MyClass : IDisposable
{
    public void Dispose()
    {
        //释放
    }
}

调用Dispose()方法:

MyClass my = new MyClass();
//代码
my.Dispose();

这种释放方式,如果在过程代码中抛出异常,Dispose()方法就没有被调用,导致内存没有被释放掉。

MyClass my = new MyClass();
try
{
    //代码
}
finally
{
    my.Dispose();
}

通过以上调用方式,可以避免过程代码抛出异常,导致内存没被释放掉。还可以使用using关键字来简化调用,效果同上面一样。

using (MyClass my = new MyClass())
{
    //代码
}

(三)不安全代码

1、用指针直接访问内存

指针只是一个以与引用相同的方式存储地址的变量。

(1)用unsafe关键字编写不安全的代码

不安全代码所使用的关键字是unsafe。

unsafe class MyClass //不安全类
{
    unsafe public string Name { get; set; }//不安全属性
    unsafe void SayHi()//不安全方法
    {
        Console.WriteLine("Hi!"+ Name);
    }
    void SayBay()
    {
        unsafe int* pAge;//不安全局部变量需要在不安全方法里,这里会报错
        Console.WriteLine("Bye!" + pAge);
    }
}

(2)指针的语法

把代码块标记为unsafe后,使用以下语法声明指针:

int* age;

声明指针类型的变量后,就可以用与一般变量相同的方式使用它们,但首先需要学习另外两个运算符:&表示“取地址”,*表示“获取地址的内容”。

int x = 10;
int* pX = &x;
int y = *pX;

可以把指针声明为任意一种值类型。

(3)将指针强制转换为整数类型

由于指针实际上存储了一个表示地址的整数,因此任何指针中的地址都可以和任何整数类型之间相互转换。

int x = 100;
ulong* pY = (ulong*)x;

需要注意的是,在32位系统上,一个地址占4个字节,把指针转换为非uint、long或ulong时可能会导致溢出错误,64位系统一个地址占8个字节,把指针转换为非ulong时会导致溢出错误。还要注意,指针的溢出无法通过checked关键字来检查。因为.NET运行库假定,如果使用指针就知道自己在做什么,不必担心溢出。

(4)指针类型之间的强制转换

byte b = 10;
byte* pB = &b;
double* pD = (double*)pB;

(5)void指针

byte b = 10;
byte* pB = &b;
void* pV = (void*)pB;

void指针的主要作用是调用需要void*参数的API函数。

(6)指针算术的运算

可以给指针加减整数。给类型为T的指针加上数值X,其中指针的值为P,则得到的结果时P+X*(sizeof(T))。

byte b = 10;
byte* pB = &b;
pB--;

如果两个指针类型相同,则可以把一个指针减去另一个指针,结果时一个long类型值为两差值除类型所占字节数整除的结果

byte b1 = 10;
byte* pB1 = &b1;
byte b2= 11;
byte* pB2 = &b2;
long l = pB1 - pB2;

(7)sizeof运算符

使用sizeof运算符,它的参数是数据类型的名称,返回该类型所占字符数。

int x = sizeof(int);//4

(8)结构指针:指针成员访问运算符

结构指针的工作方式与预定义值类型的指针的工作方式完全相同。但是这有一个条件:结构不能包含任何引用类型,因为指针不能指向任何引用类型。

MyStruct* pStruct;
MyStruct myStruct=new  MyStruct();
pStruct= &myStruct;
//通过指针访问结构成员值
(*pStruct).X = 4;
//另一种语法
*pStruct->X = 4;

(9)类成员指针

不能创建指向类的指针,这是因为垃圾回收期不维护关于指针的任何信息,只维护关于引用的信息,而在垃圾回收的过程中堆会被移动,这样就会导致指针指向错误,为了解决这个问题需要使用fixed关键字,这样告诉垃圾回收器,不移动这些对象。

MyClass myClass = new MyClass();
fixed (double* pX = &(myClass.X))//多个这样的指针可以在代码块之前放置多条
fixed (long* pX = &(myClass.Y), pZ = &(myClass.Z))//指针类型相同时可以在一个括号内声明
{
    fixed (long* pW&(myClass.W))//嵌套声明
    { }
}

2、使用指针优化性能

1、创建基于栈的数组

指针的一个主要应用领域:在栈中创建高性能、低系统开销的数组。为了创建一个高性能数组,需要使用另一个关键字:stackalloc。stackalloc命令提示.NET运行库在栈上分配一定量的内存(数据类型所占字节数乘以项数)。在调用stackalloc命令时,需要提供要存储的数据类型(必须是值类型)、需要存储的数据项数。

decimal* pDecimal = stackalloc decimal[10];

项数还可以是一个变量:

int size = 5;
decimal* pDecimal = stackalloc decimal[size];

stackalloc总是返回分配数据类型的指针,它指向新分配的内存块的顶部。

要访问数组的下一个元素,可以使用指针算法。用表达式*(pDecimal+X)访问数组中下标为X的元素。

*pDecimal = 1;//数组第1项
*(pDecimal + 1) = 2;//数组第2项
C#还定义了另一种方法来访问数组,与正常的数组访问方式相同。
pDecimal[0] = 1;//等同与*pDecimal = 1;
pDecimal[1] = 2; //等同与*(pDecimal + 1) = 2;

需要注意的是,当使用指针时编译器无法检查变量,这个时候当访问项数超出分配的项数时会在运行时抛出异常。

pDecimal[20] = 21;

使用指针在获得高性能的同时,也会付出一些代价:需要确保自己知道在做什么,否则就会抛出非常古怪的运行错误。

时间: 2024-11-05 04:52:52

【读书笔记】C#高级编程 第十四章 内存管理和指针的相关文章

读书笔记 - js高级程序设计 - 第十五章 使用Canvas绘图

读书笔记 - js高级程序设计 - 第十三章 事件 canvas 具备绘图能力的2D上下文 及文本API 很多浏览器对WebGL的3D上下文支持还不够好 有时候即使浏览器支持,操作系统如果缺缺乏必要的绘图驱动程序,则浏览器即使支持了也没用   <canvas> var drawing = document.getElementById("drawing"); if( drawing.getContext ){ drawing.getContext("2d"

R in action读书笔记(19)第十四章 主成分和因子分析

第十四章:主成分和因子分析 本章内容 主成分分析 探索性因子分析 其他潜变量模型 主成分分析(PCA)是一种数据降维技巧,它能将大量相关变量转化为一组很少的不相关变量,这些无关变量称为主成分.探索性因子分析(EFA)是一系列用来发现一组变量的潜在结构的方法.它通过寻找一组更小的.潜在的或隐藏的结构来解释已观测到的.显式的变量间的关系. PCA与EFA模型间的区别 主成分(PC1和PC2)是观测变量(X1到X5)的线性组合.形成线性组合的权重都是通过最大化各主成分所解释的方差来获得,同时还要保证个

【读书笔记】C#高级编程 第二十四章 文件和注册表操作

(一)文件和注册表 对于文件系统操作,相关的类几乎都在System.IO名称空间中,而注册表操作由System.Win32名称空间中的类来处理. (二)管理文件系统 System.MarshalByRefObject--这是.NET类中用于远程操作的基对象类,它允许在应用程序域之间编组数据. FileSystemInfo--这是表示任何文件系统对象的基类. FileInfo和File--这些类表示文件系统上的文件. DirectoryInfo和Directory--这些类表示文件系统上的文件夹.

读书笔记 - js高级程序设计 - 第十二章 DOM2和DOM3

Node类型的变化   访问元素的样式 myDiv.style.backgroundColor = "red" myDiv.style.width = "100px" 计算的样式 记住所有计算的样式都是只读的 偏移量 offsetHeight 外边框外 offsetWidth offsetLeft  外边框外 到 左端 offsetTop  客户区的大小 clientWidth  内边框外缘 clientHeight 内边框外缘 滚动大小 scrollHeight

javascript高级程序设计 第十四章--表单脚本

javascript高级程序设计 第十四章--表单脚本 在HTML中表单由<form>元素表示,在js中表单对应的是HTMLFormElement类型,这个类型也有很多属性和方法:取得表单元素的引用还是为它添加id特性,用DOM操作来获取表单元素:提交表单:把<input>或<button>元素的type特性设置为"submit",图像按钮把<input>元素的type特性设置为"image",也可以调用submit(

R in action读书笔记(16)第十二章 重抽样与自助法之 置换检验

第十二章:重抽样与自助法 本章,我们将探究两种应用广泛的依据随机化思想的统计方法:置换检验和自助法 12.1 置换检验 置换检验,也称随机化检验或重随机化检验. 有两种处理条件的实验,十个受试者已经被随机分配到其中一种条件(A或B)中,相应的结果变量(score)也已经被记录.实验结果如下: 如果两种处理方式真的等价,那么分配给观测得分的标签(A处理或B处理)便是任意的.为检验两种处理方式的差异,我们可遵循如下步骤: (1) 与参数方法类似,计算观测数据的t统计量,称为t0: (2) 将10个得

【读书笔记】C#高级编程 第十二章 动态语言扩展

(一)DLR C#4的动态功能是Dynamic Language Runtime(动态语言运行时,DLR)的一部分.DLR是添加到CLR的一系列服务. (二)dynamic类型 dynamic类型允许编写忽略编译期间的类型检查的代码.编译器假定,给dynamic类型的对象定义的任何操作都是有效的,在运行之前编译器不会检测是否存在错误. 例子: dynamic person = "人"; string firstName = person.FirstName; 这两行代码能够通过编译器编

【读书笔记】C#高级编程 第十六章 错误和异常

(一)简介 错误的出现并不总是编写应用程序的人的原因,有时应用程序会因为应用程序的最终用户引发或运行代码的环境而发生错误.C#提供了异常处理机制来处理错误. (二)异常类 在C#中,但刚出现某个特殊的异常错误条件时,就会创建(或抛出)一个异常对象.一般情况下异常没有特定的名称空间,异常类应放在生成异常的类所在的名称空间. (三)捕获异常 .NET Framework提供了大量的预定义基类异常对象.为了在C#代码中处理可能的错误情况,一般要把程序的相关部分分成3种不同类型的代码块. try块包含的

【读书笔记】C#高级编程 第二十二章 安全性

(一)身份验证和授权 安全性的两个基本支柱是身份验证和授权.身份验证是标识用户的过程,授权在验证了所标识用户是否可以访问特性资源之后进行的. 1.标识和Principal 使用标识可以验证运行应用程序的用户.Principal是一个包含用户的标识和用户所属角色的对象. AppDomain.CurrentDomain.SetPrincipalPolicy(PrincipalPolicy.WindowsPrincipal); var principal = WindowsPrincipal.Curr