【读书笔记】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;
使用指针在获得高性能的同时,也会付出一些代价:需要确保自己知道在做什么,否则就会抛出非常古怪的运行错误。
【读书笔记】C#高级编程 第十四章 内存管理和指针的更多相关文章
- C#高级编程9 第14章 内存管理和指针
C#高级编程9 内存管理和指针 后台内存管理 1) 值数据类型 在处理器的虚拟内存中有一个区域,称为栈,栈存储变量的浅副本数据,通过进入变量的作用域划分区域,通过离开变量的作用域释放. 栈的指针指向栈 ...
- 读书笔记 - js高级程序设计 - 第十五章 使用Canvas绘图
读书笔记 - js高级程序设计 - 第十三章 事件 canvas 具备绘图能力的2D上下文 及文本API 很多浏览器对WebGL的3D上下文支持还不够好 有时候即使浏览器支持,操作系统如果缺缺 ...
- R in action读书笔记(19)第十四章 主成分和因子分析
第十四章:主成分和因子分析 本章内容 主成分分析 探索性因子分析 其他潜变量模型 主成分分析(PCA)是一种数据降维技巧,它能将大量相关变量转化为一组很少的不相关变量,这些无关变量称为主成分.探索性因 ...
- 【读书笔记】C#高级编程 第二十四章 文件和注册表操作
(一)文件和注册表 对于文件系统操作,相关的类几乎都在System.IO名称空间中,而注册表操作由System.Win32名称空间中的类来处理. (二)管理文件系统 System.MarshalByR ...
- 读书笔记 - js高级程序设计 - 第十二章 DOM2和DOM3
Node类型的变化 访问元素的样式 myDiv.style.backgroundColor = "red" myDiv.style.width = "100px& ...
- 【读书笔记】自然语言处理综述 -- 第四章 -- N元语法
第四章 N元语法 本章开篇的两句话很有意思,代表了当时两个学派的思想和矛盾. 一句是"有史以来最伟大的语言学家"乔姆斯基说的:"句子的概率,在任何已知的对于这个术语的解释 ...
- R in action读书笔记(16)第十二章 重抽样与自助法之 置换检验
第十二章:重抽样与自助法 本章,我们将探究两种应用广泛的依据随机化思想的统计方法:置换检验和自助法 12.1 置换检验 置换检验,也称随机化检验或重随机化检验. 有两种处理条件的实验,十个受试者已经被 ...
- 【读书笔记】C#高级编程 第十九章 程序集
(一)程序集的含义 程序集是.NET用于部署和配置单元的术语. .NET应用程序包含一个或多个程序集.通常扩展名是EXE或DLL的.NET可执行程序称为程序集. 程序集是自我描述的安装单元,由一个或多 ...
- 【读书笔记】C#高级编程 第二十二章 安全性
(一)身份验证和授权 安全性的两个基本支柱是身份验证和授权.身份验证是标识用户的过程,授权在验证了所标识用户是否可以访问特性资源之后进行的. 1.标识和Principal 使用标识可以验证运行应用程序 ...
随机推荐
- Linux文件的通配符
通配符的作用:匹配文件名 常见的通配符: *:表示任意个字符(不包括隐藏文件) ?:单个任意字符(中文也算一个字符) []:表示匹配一范围或者其中一个 表示匹配范围: [a-z] --- 不但包括了小 ...
- 在.NET 6.0上使用Kestrel配置和自定义HTTPS
大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 本章是<定制ASP NET 6.0框架系列文章>的第四篇.在本章,我们 ...
- java 改变图片的DPI
代码如下: public class test01 { private static int DPI = 300; public static void main(String[] args) { S ...
- Linux 批量杀死进程(详细版本)
使用场景 当程序中有使用到多进程且进程数较多的情况,如下图,且需要通过控制台杀死所有的 GSM_run.py 的进程时,利用 kill 命令一个一个的去结束进程是及其耗时且繁琐的,这时就需要我们的ki ...
- redis 集群 slots are covered by nodes.
原因数据数据损坏.需要修复 1.检测 redis-cli --cluster check 127.0.0.1:7000 2.检测结果 slots are covered by nodes3.进行修复 ...
- 1_day01_java入门
java入门 学习目标: 1.熟悉计算机编程语言 2.熟练掌握java特点 3.熟练配置java开发环境 4.熟练编写入门程序 5.熟练编写注释信息 一.计算机语言 1.1 什么是编程语言 计算机语言 ...
- Techempower web框架性能测试第21轮结果发布--asp.net core继续前进
废话不说,直接上结果: Round 21 results - TechEmpower Framework Benchmarks Techempower benchmark是包含范围最广泛的web框架性 ...
- Flink Window&Time 原理
Flink 中可以使用一套 API 完成对有界数据集以及无界数据的统一处理,而无界数据集的处理一般会伴随着对某些固定时间间隔的数据聚合处理.比如:每五分钟统计一次系统活跃用户.每十秒更新热搜榜单等等 ...
- 2022-7-11第五组 pan小堂 js基础
##为何学习 JavaScript? ###JavaScript 是 web 开发者必学的三种语言之一: HTML 定义网页的内容 CSS 规定网页的布局 JavaScript 对网页行为进行编程 在 ...
- python 上下文管理(with、contextmanager)
简介 python中存在这么一个非常好用的东西,能够帮助我们更好的管理上下文,这种东西就成为上下文管理器,例如我们最常见的with open,在读取文件时可以很方便的进行文件描述符和文件对象的打开和关 ...