C#高级编程七十三天----浅析C#中内存管理

2023-11-21 10:48

本文主要是介绍C#高级编程七十三天----浅析C#中内存管理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

浅析C#中内存管理

前言:个人觉得C#吸收了各种语言的优点,可谓集大成者,但是不知但,这种集所有语言于一身的情况是好是坏.C#编程的一个优点就是程序员不需要关心具体的内存管理,尤其是垃圾收集器会处理所有的内存清理工作.虽然不比手工管理内存,但是如果要编写高质量的代码,还是要理解后台发生的情况,理解C#的内存管理.

用户可以得到像C++语言那样的效率,而不需要考虑像在C++中那样内存管理工作的复杂性.

 

 

 

注意:这一章的许多内容,可以说全部,都是没有经过事实验证的.您应把这一届看做是一般规则的简化向导,而不是实现的确切说明.

 

 

首先C#将数据分为两种:值数据类型和引用数据类型,这两种数据类型存储在内存中的不同的地方:值数据类型存储在堆栈中,而引用类型存储在内存的托管堆中.

 

 

一.内存简介

windows使用一个系统:虚拟寻址系统.这个系统的作用是将程序可用的内存地址映射到硬件内存中的实际地址.其实际结果就是32位的机子上每个进程都可以使用4GB的内存,当然,64位的机器4,这个数字就更大了.4G的内存实际上包含了程序的所有的部分,可执行代码,DLL以及程序运行时使用的所有变量的内容.这个4GB内存称为虚拟地址空间或虚拟内存.为方便,着了成为内存.

4GB中的每个存储单元都是从零开始向上存储的.要访问存储在内存中的某个空间中的值,就必须提供表示该存储单元的一个数字.在高级编程语言中,比哪一期的一个重要作用就是负责将人们可以理解的变量名称编程处理器可以理解的内存地址.

 

二.堆栈

在内存中,有一个区域称为堆栈,存储对象.

对象成员的值数据类型调用方法时,传递给所有方法的参数的副本.注意:调用方法时,堆栈存储的是所有参数的副本,因此,经值类型A传递给函数,A的值是不会变化的.当然,引用类型时会变化的,因为堆栈中存储的是引用类型的地址.

 

案例:

{

int a;

//do domething

 

{

int b;

//do domething

}

}

 

分析:首先声明a,在内部的代码块中声明b,然后内部的代码块终止,b就出了作用域,最后a出作用域.所以b的声明周期总是包含在a的声明周期内,在释放变量的时候,其顺序总是和分配内存的顺序相反.:变量的生存周期都是嵌套的.这就是堆栈的工作方式.

 

三.托管堆

堆栈具有相当高的性能,但是变量的生命周期必须是嵌套的,这个要求在有的时候过于苛求.我们希望有一种别的方法来分配内存,存储一些数据,并在方法退出的很长一段时间内,这些户数仍然是可用的,这时就可以使用托管堆.

 

托管堆(简称堆)是内存中的另一个区域,我们仍然用一个例子来说明堆的工作方式:

{

Customer customer1;

customer1=new Customer();

Customer customer2=new Customer();

//do domething

}

 

分析:首先,声明一个Customer:customer1,在堆栈上给这个引用分配空间.注意:仅仅是给这个引用分配存储空间,并不是实际的Customer对象.customer1占用4个字节的空间(32位机),来表示Customer对象在内存中的地址.

 

然后.执行第二行代码,完成以下操作:

在堆上分配存储空间,用来存储CUstomer对象,注意,这里是Customer对象.

将变量customer1的值设为分配给Customer对象的内存地址从这个例子中可以看出,建立引用类型的变量的过程要比建立值类型变量的过程复杂,且不能避免性能的降低.但是,我们可以将一个引用变量的值赋给另一个引用变量,当一个变量出了作用域时,他会从对战中删除,但是对象的数据仍然保留在内存中,直到程序停止.

 

这样,我们将一个引用变量A传递给函数时,仅仅是将变量A的引用传递给了函数,:仅仅是在对站上分配内存,即变量B两者指向同一个内存地址.因此,当变量发生变化时,变量A也会发生变化.

 

 

四.拆装箱

 

拆装箱就是值类型和引用类型的项目转化,装箱可以将值类型转化为引用类型,拆箱的作用恰好相反,经引用类型转化为值类型.

 

 

 

 

.垃圾回收

 

一般情况下. .NET运行库会在认为需要的时候运行垃圾收集器来释放托管资源,这在大多数情况下是足够的,就是说我们没有必要去关心内存.但在有的情况下,我们会强制垃圾回收器在代码的某个地方运行,释放内存.这就用到了System.GC.Collect;System.GV表示一个垃圾回收期.这种情况很少见,例如,代码中大量的对象刚刚停止引用,就是和调用垃圾收集器.

 

 

 

这篇关于C#高级编程七十三天----浅析C#中内存管理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/401861

相关文章

C#中Guid类使用小结

《C#中Guid类使用小结》本文主要介绍了C#中Guid类用于生成和操作128位的唯一标识符,用于数据库主键及分布式系统,支持通过NewGuid、Parse等方法生成,感兴趣的可以了解一下... 目录前言一、什么是 Guid二、生成 Guid1. 使用 Guid.NewGuid() 方法2. 从字符串创建

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri

Python中你不知道的gzip高级用法分享

《Python中你不知道的gzip高级用法分享》在当今大数据时代,数据存储和传输成本已成为每个开发者必须考虑的问题,Python内置的gzip模块提供了一种简单高效的解决方案,下面小编就来和大家详细讲... 目录前言:为什么数据压缩如此重要1. gzip 模块基础介绍2. 基本压缩与解压缩操作2.1 压缩文

C# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

Java中的for循环高级用法

《Java中的for循环高级用法》本文系统解析Java中传统、增强型for循环、StreamAPI及并行流的实现原理与性能差异,并通过大量代码示例展示实际开发中的最佳实践,感兴趣的朋友一起看看吧... 目录前言一、基础篇:传统for循环1.1 标准语法结构1.2 典型应用场景二、进阶篇:增强型for循环2.

使用Python进行GRPC和Dubbo协议的高级测试

《使用Python进行GRPC和Dubbo协议的高级测试》GRPC(GoogleRemoteProcedureCall)是一种高性能、开源的远程过程调用(RPC)框架,Dubbo是一种高性能的分布式服... 目录01 GRPC测试安装gRPC编写.proto文件实现服务02 Dubbo测试1. 安装Dubb

浅析如何保证MySQL与Redis数据一致性

《浅析如何保证MySQL与Redis数据一致性》在互联网应用中,MySQL作为持久化存储引擎,Redis作为高性能缓存层,两者的组合能有效提升系统性能,下面我们来看看如何保证两者的数据一致性吧... 目录一、数据不一致性的根源1.1 典型不一致场景1.2 关键矛盾点二、一致性保障策略2.1 基础策略:更新数