【转】深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第五节 引用类型复制问题及用克隆接口ICloneable修复

前言

 

虽然在.Net Framework 中我们不必考虑内在管理和垃圾回收(GC),但是为了优化应用程序性能我们始终需要了解内存管理和垃圾回收(GC)。另外,了解内存管理可以帮助我们理解在每一个程序中定义的每一个变量是怎样工作的。

 

简介

这一节我们将介绍引用类型变量在堆中存储时会产生的问题,同时介绍怎么样使用克隆接口ICloneable去修复这种问题。

 

 

复制不仅仅是复制

 

为了更清晰的阐述这个问题,让我们测试一下在堆中存储值类型变量和引用类型变量时会产生的不同情况。

 

值类型测试

 

首先,我们看一下值类型。下面是一个类和一个结构类型(值类型),Dude类包含一个Name元素和两个Shoe元素。我们有一个CopyDude()方法用来复制生成新Dude。

 
  1. public struct Shoe{

  2. public string Color;

  3. }

  4.  
  5. public class Dude

  6. {

  7. public string Name;

  8. public Shoe RightShoe;

  9. public Shoe LeftShoe;

  10.  
  11. public Dude CopyDude()

  12. {

  13. Dude newPerson = new Dude();

  14. newPerson.Name = Name;

  15. newPerson.LeftShoe = LeftShoe;

  16. newPerson.RightShoe = RightShoe;

  17.  
  18. return newPerson;

  19. }

  20.  
  21. public override string ToString()

  22. {

  23. return (Name + " : Dude!, I have a " + RightShoe.Color +

  24. " shoe on my right foot, and a " +

  25. LeftShoe.Color + " on my left foot.");

  26. }

  27.  
  28. }


Dude类是一个复杂类型,因为值 类型结构Shoe是它的成员, 它们都将存储在堆中。

 

 

当我们执行下面的方法时:

 
  1. public static void Main()

  2. {

  3. Class1 pgm = new Class1();

  4.  
  5. Dude Bill = new Dude();

  6. Bill.Name = "Bill";

  7. Bill.LeftShoe = new Shoe();

  8. Bill.RightShoe = new Shoe();

  9. Bill.LeftShoe.Color = Bill.RightShoe.Color = "Blue";

  10.  
  11. Dude Ted = Bill.CopyDude();

  12. Ted.Name = "Ted";

  13. Ted.LeftShoe.Color = Ted.RightShoe.Color = "Red";

  14.  
  15. Console.WriteLine(Bill.ToString());

  16. Console.WriteLine(Ted.ToString());

  17.  
  18. }


我们得到了期望的结果:

Bill : Dude!, I have a Blue shoe on my right foot, and a Blue on my left foot.
Ted : Dude!, I have a Red shoe on my right foot, and a Red on my left foot.

如果我们把Shoe换成引用类型呢?

 

引用类型测试

 

当我们把Shoe改成引用类型时,问题就产生了。

 
  1. public class Shoe{

  2. public string Color;

  3. }


执行同样上面的Main()方法,结果改变了,如下:

Bill : Dude!, I have a Red shoe on my right foot, and a Red on my left foot
Ted : Dude!, I have a Red shoe on my right foot, and a Red on my left foot


这并不是我们期望的结果。很明显,出错了!看下面的图解:

 

因为现在Shoe是引用类型而不是值类型,当我们进行复制时仅是复制了指针,我们并没有复制指针真正对应的对象。这就需要我们做一些额外的工作使引用类型Shoe像值类型一样工作。

很幸运,我们有一个接口可以帮我们实现:ICloneable。当Dude类实现它时,我们会声明一个Clone()方法用来产生新的Dude复制类。(译外话:复制类及其成员跟原始类不产生任何重叠,即我们所说的深复制)   看下面代码:

 
  1. ICloneable consists of one method: Clone()

  2.  
  3. public object Clone()

  4. {

  5.  
  6. }

  7.  
  8. Here's how we'll implement it in the Shoe class:

  9.  
  10. public class Shoe : ICloneable

  11. {

  12. public string Color;

  13. #region ICloneable Members

  14.  
  15. public object Clone()

  16. {

  17. Shoe newShoe = new Shoe();

  18. newShoe.Color = Color.Clone() as string;

  19. return newShoe;

  20. }

  21.  
  22. #endregion

  23. }


在Clone()方法里,我们创建了一个新的Shoe,克隆所有引用类型变量,复制所有值类型变量,最后返回新的对象Shoe。有些既有类已经实现了ICloneable,我们直接使用即可,如String。因此,我们直接使用Color.Clone()。因为Clone()返回object对象,我们需要进行一下类型转换。

下一步,我们在CopyDude()方法里,用克隆Clone()代替复制:

 
  1. public Dude CopyDude()

  2. {

  3. Dude newPerson = new Dude();

  4. newPerson.Name = Name;

  5. newPerson.LeftShoe = LeftShoe.Clone() as Shoe;

  6. newPerson.RightShoe = RightShoe.Clone() as Shoe;

  7.  
  8. return newPerson;

  9. }


再次执行主方法Main():

 
  1. public static void Main()

  2. {

  3. Class1 pgm = new Class1();

  4.  
  5. Dude Bill = new Dude();

  6. Bill.Name = "Bill";

  7. Bill.LeftShoe = new Shoe();

  8. Bill.RightShoe = new Shoe();

  9. Bill.LeftShoe.Color = Bill.RightShoe.Color = "Blue";

  10.  
  11. Dude Ted = Bill.CopyDude();

  12. Ted.Name = "Ted";

  13. Ted.LeftShoe.Color = Ted.RightShoe.Color = "Red";

  14.  
  15. Console.WriteLine(Bill.ToString());

  16. Console.WriteLine(Ted.ToString());

  17.  
  18. }


我们得到了期望的结果:

 

Bill : Dude!, I have a Blue shoe on my right foot, and a Blue on my left foot
Ted : Dude!, I have a Red shoe on my right foot, and a Red on my left foot


下面是图解:

 

整理我们的代码

 

在实践中,我们是希望克隆引用类型并复制值类型的。这会让你回避很多不易察觉的错误,就像上面演示的一样。这种错误有时不易被调试出来,会让你很头疼。

 

因此,为了减轻头疼,让我们更进一步清理上面的代码。我们让Dude类实现IConeable代替使用CopyDude()方法:

 
  1. public class Dude: ICloneable

  2. {

  3. public string Name;

  4. public Shoe RightShoe;

  5. public Shoe LeftShoe;

  6.  
  7. public override string ToString()

  8. {

  9. return (Name + " : Dude!, I have a " + RightShoe.Color +

  10. " shoe on my right foot, and a " +

  11. LeftShoe.Color + " on my left foot.");

  12. }

  13. #region ICloneable Members

  14.  
  15. public object Clone()

  16. {

  17. Dude newPerson = new Dude();

  18. newPerson.Name = Name.Clone() as string;

  19. newPerson.LeftShoe = LeftShoe.Clone() as Shoe;

  20. newPerson.RightShoe = RightShoe.Clone() as Shoe;

  21.  
  22. return newPerson;

  23. }

  24.  
  25. #endregion

  26. }


在主方法Main()使用Dude.Clone():

 
  1. public static void Main()

  2. {

  3. Class1 pgm = new Class1();

  4.  
  5. Dude Bill = new Dude();

  6. Bill.Name = "Bill";

  7. Bill.LeftShoe = new Shoe();

  8. Bill.RightShoe = new Shoe();

  9. Bill.LeftShoe.Color = Bill.RightShoe.Color = "Blue";

  10.  
  11. Dude Ted = Bill.Clone() as Dude;

  12. Ted.Name = "Ted";

  13. Ted.LeftShoe.Color = Ted.RightShoe.Color = "Red";

  14.  
  15. Console.WriteLine(Bill.ToString());

  16. Console.WriteLine(Ted.ToString());

  17.  
  18. }


最后得到期望的结果:

Bill : Dude!, I have a Blue shoe on my right foot, and a Blue on my left foot.
Ted : Dude!, I have a Red shoe on my right foot, and a Red on my left foot.

 

特殊引用类型String

 

在C#中有趣的是,当 System.String 使用操作符“=”时,实际上是进行了克隆(深复制)。你不必担心你只是在操作一个指针,它会在内存中创建一个新的对象。但是,你一定要注意内存的占用问题(译外话:比如为什么在一定情况下我们使用StringBuilder代替String+String+String+String...前者速度稍慢初始化耗多点内存但在大字符串操作上节省内存,后者速度稍快初始化简单但在大字符串操作上耗内存)。如果我们回头去看上面的图解中,你会发现Stirng类型在图中并不是一个针指向另一个内存对象,而是为了尽可能的简单,把它当成值类型来演示了。

 

总结

 

在实际工作中,当我们需要复制引用类型变量时,我们最好让它实现ICloneable接口。这样可以让引用类型模仿值类型的使用,从而防止意外的错误产生。你可以看到,慎重得理不同的类型非常重要,因为值类型和引用类型在内存中的分配是不同的。

 

 

翻译: http://www.c-sharpcorner.com/UploadFile/rmcochran/chsarp_memory401152006094206AM/chsarp_memory4.aspx

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436776.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux virt java_Linux下Java环境安装

本节主要讲解Linux(Centos 6.5)下Java环境的安装1. 卸载机器上默认安装的JDK在Linux环境下一般会默认安装jdk,为了自己项目的开发部署,一般情况要重新装jdk,而且自己装的Jdk相对来说易控制版本,稳定性更高。所以以下是我卸载预装J…

【转】深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第六节 理解垃圾回收GC,提搞程序性能****

前言 虽然在.Net Framework 中我们不必考虑内在管理和垃圾回收(GC),但是为了优化应用程序性能我们始终需要了解内存管理和垃圾回收(GC)。另外,了解内存管理可以帮助我们理解在每一个程序中定义的每一个变量是怎样工作的。 简介 这一节我们将介绍垃圾回…

java下拉列表 动态_【示例】教你简单用Java写一个动态更新的下拉列表(无数据库)...

动态更新下拉列表varxmlHttp;functioncreatXMLHttpRequest(){if(window.ActiveXObject){xmlHttpnewActiveXObject("Microsoft.XMLHTTP");}else if(window.XMLHttpRequest){xmlHttpnewXMLHttpRequest();}}functionupdateSelect(){varselecteddocument.all.slt1.value;…

【转】分布式事务的常见解决方案

一、事务起步 1. 什么是事务 事务这种东西大家都耳熟能详了,通常指由一组操作组成的一个工作单元,这一整个组合要么全部成功,要么全部失败。 2. 本地事务 在计算机系统中,更多的是通过关系型数据库来控制事务,这是…

深入解析java web_java进阶--深入分析java Web

第一章: 深入了解Web请求整理本书的内容与之前的采用相同的方式,主要目的还是为了可以仔细的阅读。整理自己的见解。这本书整体的感觉很好,思路很清晰,最近就发现,国人写的文字和外国译文相差很大,主要体现…

【转】修饰符new将父类中的该方法隐藏掉有什么意义 不隐藏有什么弊端

这是一个C#语法的问题。子类如果要重写父类的方法的话,virtual-override一定成对的。 子类不重写,而是创建一个属于自己的同名方法,就最好加个new。如果不加new也等于new,但是编译器都会提醒你,加个标识比较好。 区别见…

java s.charat_Java中s.charAt(index)用于提取字符串s中的特定字符操作

charAt(int index)方法是一个能够用来检索特定索引下的字符的String实例的方法.charAt()方法返回指定索引位置的char值。索引范围为0~length()-1.如: str.charAt(0)检索str中的第一个字符,str.charAt(str.length()-1)检索最后一个字符.警告:在字符串s中越界访问字符…

【转】.NET框架简介

.NET 框架是由微软开发的软件开发平台,其最主要的两个组成部分是公共语言运行时 (CLR) 和框架类库 (FCL),基础类库 (BCL)是框架类库的一个子集。 .NET 框架简介 下图展示了 .NET 框架的主要结构。 其中,最下层的无疑就是操作系统了。 在 …

eclipse java maven_java – 非常轻量级的Eclipse-Maven集成 – 仅...

我找到了一个非常适合我所描述的需求/用例的解决方案:1.我在Eclipse项目根目录中创建了非常小的pom文件:4.0.0com.sobczyk.piotrmvn-eclipse-test1.0.0srcbinlog4jlog4j1.2.16这个.pom文件由三部分组成:>必需的Maven东西,即. modelVersion…

【转】Path.Combine (合并两个路径字符串)方法的一些使用细节

System.IO.Path.Combine 简单来说,就是合并两个路径字符串。 比如下面这个调用, Path.Combine(“C:\11”,“aa.txt”) 返回的字符串路径为: C:\11\aa.txt 这个方法的声明如下: public static string Combine ( string path1, st…

java 正则表达式 开头 结尾_Java-正则表达式匹配 #开头结尾

引包import java.util.regex.Matcher;import java.util.regex.Pattern;方法1:// 匹配 #开头结尾中,#以及中间得字符串 #xxx 替换为 "" #123 匹配#123String tableModle "#123#2#3#4";Pattern pPattern.compile("…

java比赛题目_【蓝桥杯2016第七届比赛题目】JAVA A组

1 煤球数目有一堆煤球,堆成三角棱锥形。具体:第一层放1个,第二层3个(排列成三角形),第三层6个(排列成三角形),第四层10个(排列成三角形),....如果一共有100层,共有多少个煤球?请填表…

【转】C#技术漫谈之垃圾回收机制(GC)

摘要:今天我们漫谈C#中的垃圾回收机制,本文将从垃圾回收机制的原理讲起,希望对大家有所帮助。 GC的前世与今生 虽然本文是以.NET作为目标来讲述GC,但是GC的概念并非才诞生不久。早在1958年,由鼎鼎大名的图林奖得主John…

【转】git hub 使用小结

【转自:https://blog.csdn.net/yj310873325/article/details/79255134】 1.创建账号: https://github.com/ 2.下载客户端:https://git-scm.com/download 这是命令行模式,用着比较舒服,不是github的客户端,一路默认安…

java 服务降级_微服务的降级学习

参考博客:https://blog.csdn.net/glory1234work2115/article/details/51626322https://blog.csdn.net/xiaofei0859/article/details/79180406什么是降级?1.开关先讲一下开关的由来,例如京东在6月18日做店庆促销活动,在交易下单环节…

【转】细说.NET 中的多线程 (一 概念)

为什么使用多线程 1.使用户界面能够及时响应用户的输入 当某个应用程序在进行大量运算时候,为了保证应用程序能够随时响应客户的输入,这个时候我们往往需要让大量运算和响应用户输入这两个行为在不同的线程中进行。 2.效率原因 应用程序经常需要等待一…

java内存加载dll_jacob调用dll控件,是否要执行内存释放,具体方法怎么写

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼程序用jacob调用dll控件,执行考勤的一些数据获取,可是当很多人同时操作时,会把tomcat关掉,然后产生一个错误文件,部分内容如下:## An unexpected error has been detected by HotSpot…

【转】细说.NET中的多线程 (二 线程池)

上一章我们了解到,由于线程的创建,销毁都是需要耗费大量资源和时间的,开发者应该非常节约的使用线程资源。最好的办法是使用线程池,线程池能够避免当前进程中大量的线程导致操作系统不停的进行线程切换,当线程数量到达…

java第二章_零基础学Java第二章

一、第一个代码案例1.1. HelloWorld案例1.1.1 代码执行流程我们写的代码都将以.java开头的文件保存,经过类编译器编译成.class的字节码文件,然后通过解释器翻译与机器交流1.1.1 代码执行流程1. 编写代码步骤首先定义一个类:public class 类名…

【转】细说.NET中的多线程 (三 使用Task)

上一节我们介绍了线程池相关的概念以及用法。我们可以发现ThreadPool. QueueUserWorkItem是一种起了线程之后就不管了的做法。但是实际应用过程,我们往往会有更多的需求,比如如何更简单的知道线程池里面的某些线程什么时候结束,线程结束后如何…