当前位置: 首页 > news >正文

c#中值类型和引用类型

在C#中有两种类型的数据,一种是值类型数据,一种是引用类型数据。在编码的时候区分这两种类型数据,可以避免一些细小的编码错误。

  首先说说什么类型是值类型,例如:int、float、bool之类的基础类型,以及用struct定义的类型,如:DateTime。除此外,如string,数组,以及用class定义的类型等都是引用类型。对于C#来说,很难罗列出所有类型进行一一分别,这需要自己在编码过程中进行分析总结。

  为了更好地说明两种类型之间的区别,借用如下的表格来说明。
值类型引用类型
内存分配地点分配在栈中分配在堆中
效率效率高,不需要地址转换效率低,需要进行地址转换
内存回收使用完后,立即回收使用完后,不是立即回收,等待GC回收
赋值操作进行复制,创建一个同值新对象只是对原有对象的引用
函数参数与返回值是对象的复制是原有对象的引用,并不产生新的对象
类型扩展不易扩展容易扩展,方便与类型扩展

  通过如上细致对比,大家对于值类型和引用类型有个清楚的概念。

  不过,无论是对于值类型还是引用类型来说,对于其作为函数参数或者返回值的时候,都是容易犯错误的地方。

  对于值类型来说,当其作为函数参数的时候,希望在函数中被修改,那么直接如下操作是不能被修改的。

public void Increment( int i )
{
  i++;
}

  要想在函数中对传进去的参数做真正的修改,需要借助于ref这个关键字,那么正确的形式如下。

public void Increment( ref int i )
{
 i++;
}

  也就是说,如果需要在函数中对值类型参数进行修改,需要用ref或者out进行标识才能真正实现。

  而对于引用类型来说,当其作为函数参数的时候,它所遇到的情况恰恰与值类型相反,即不希望在函数中被修改,举例如下。

public void AddValue( MyType typValue )
{
 typValue.Count = typValue.Count + 15;
}

  由于对于引用类型对象来说,其的赋值操作只是对原有对象的引用,因此在函数对其修改,实际上是直接修改了原有对象数据,这是很多情况不希望发生的(这里例如对数组或者DataTable操作这类)。

  为了防止这种事发生,需要给此类型提供clone函数。例如对于如上的类型,可以入下实现。

public class MyType:ICloneable
{
 private int nCount = 0;
 public int Count
 {
  set{ nCount = value;}
  get{ return nCount;}
 }
 public MyType()
 {}
 public MyType( int Value)
 {
  nCount = Value;
 }

 #region ICloneable Members

 public object Clone()
 {
  // TODO: Add MyType.Clone implementation
  return new MyType( nCount );
 }

 #endregion
}

  那么在调用的时候,用当前的对象的clone作为参数即可。

  不过对于引用类型来说,提供一个clone函数不是一件容易的事情,尤其出现引用类型嵌套的时候,所以说去实现一个完全clone功能是件很费事又不讨好的活,这也就是在论坛中常说的深copy和浅copy的问题。话虽如此,如果对于前面所说的有个大概了解,相信实现也不是不可能。

  在C#中,尤其自己定义类型的时候,常常由于是用struct来定义还是用class来定义,即是定义一个值类型还是一个引用类型呢。在这儿给了几个判定条件,如果如下几点都满足的话,建议用struct来定义为值类型,否则用class定义为引用类型。

<!--[if !supportLists]-->1. <!--[endif]-->这个类型是否主要为了数据存储;
<!--[if !supportLists]-->2. <!--[endif]-->是否只通过属性来访问对象的数据成员;
<!--[if !supportLists]-->3. <!--[endif]-->这个类型是否不会有子类型;
<!--[if !supportLists]-->4. <!--[endif]-->在程序处理的时候不会把这个类型对象通过多态来处理。

转载于:https://www.cnblogs.com/9who/archive/2008/07/28/1254569.html

相关文章:

  • 9.4.1 使用参数来防止SQL注入
  • ASP.NET 框架 之HttpHandler
  • 判断“textbox为空”写法
  • Using System.Threading
  • Windows Server 2008 参考价格
  • 我是一个*** (九)
  • SET NOCOUNT 使用说明
  • JSP学习笔记(二十一):struts2中访问request,application对象
  • vb.net中的键盘事件处理
  • 开源与云计算
  • 我的博客也是男的(还好)
  • 转自Microsoft-DDOS处理参考-如何:强化 TCP/IP 堆栈安全
  • 普天C++笔试题
  • 在 VS 中查找替换时使用正则表达式的语法
  • 剧透和评析之車輪の国、向日葵の少女
  • [rust! #004] [译] Rust 的内置 Traits, 使用场景, 方式, 和原因
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • 3.7、@ResponseBody 和 @RestController
  • Cookie 在前端中的实践
  • Effective Java 笔记(一)
  • Idea+maven+scala构建包并在spark on yarn 运行
  • iOS高仿微信项目、阴影圆角渐变色效果、卡片动画、波浪动画、路由框架等源码...
  • JavaScript设计模式之工厂模式
  • Promise初体验
  • Python中eval与exec的使用及区别
  • SOFAMosn配置模型
  • Spring Boot快速入门(一):Hello Spring Boot
  • 发布国内首个无服务器容器服务,运维效率从未如此高效
  • 前端存储 - localStorage
  • 扫描识别控件Dynamic Web TWAIN v12.2发布,改进SSL证书
  • 吐槽Javascript系列二:数组中的splice和slice方法
  • 微信开源mars源码分析1—上层samples分析
  • 限制Java线程池运行线程以及等待线程数量的策略
  • 写给高年级小学生看的《Bash 指南》
  • 用 vue 组件自定义 v-model, 实现一个 Tab 组件。
  • 【运维趟坑回忆录】vpc迁移 - 吃螃蟹之路
  • puppet连载22:define用法
  • #{} 和 ${}区别
  • #HarmonyOS:软件安装window和mac预览Hello World
  • $NOIp2018$劝退记
  • (2)(2.4) TerraRanger Tower/Tower EVO(360度)
  • (读书笔记)Javascript高级程序设计---ECMAScript基础
  • (附源码)springboot助农电商系统 毕业设计 081919
  • (力扣)1314.矩阵区域和
  • (七)MySQL是如何将LRU链表的使用性能优化到极致的?
  • (全注解开发)学习Spring-MVC的第三天
  • (十)【Jmeter】线程(Threads(Users))之jp@gc - Stepping Thread Group (deprecated)
  • (一)WLAN定义和基本架构转
  • (终章)[图像识别]13.OpenCV案例 自定义训练集分类器物体检测
  • .bat批处理(十):从路径字符串中截取盘符、文件名、后缀名等信息
  • .helper勒索病毒的最新威胁:如何恢复您的数据?
  • .NET Core 和 .NET Framework 中的 MEF2
  • .net 程序 换成 java,NET程序员如何转行为J2EE之java基础上(9)
  • .NET/C# 将一个命令行参数字符串转换为命令行参数数组 args
  • .NET/C# 使窗口永不获得焦点