本文深入淺出的分析了ASP.NET深度復(fù)制和淺度復(fù)制。分享給大家供大家參考。具體分析如下:
之前一直沒有搞清楚深度復(fù)制和淺度復(fù)制的區(qū)別到底在哪里,今天徹底把這個(gè)東西弄懂了,寫出來與到家共勉。
如果大家不懂值類型和引用類型的區(qū)別,請(qǐng)先看https://www.jb51.net/article/57471.htm,本來想自己寫的,但剛好看到了這篇寫的非常全面,就不自己寫了。
大家都知道Object是所有類共同的基類,其有個(gè)方法是MemberwiseClone(),其用途為
![](http://img.jbzj.com/file_images/article/201411/20141117102815085.png?20141017102917)
我們可以通過這個(gè)方法來達(dá)到淺度復(fù)制的效果。
下面我們通過一個(gè)例子來闡述一下淺度復(fù)制,其與深度復(fù)制的區(qū)別在什么地方:
復(fù)制代碼 代碼如下:
public class Content
{
public int Val;
}
public class Cloner
{
public Content MyContent = new Content();
public Cloner(int newVal)
{
MyContent.Val = newVal;
}
public object GetCopy()
{
return MemberwiseClone();
}
}
在這里我們有兩個(gè)類,一個(gè)Content類,只有一個(gè)為值類型int的Val,還有個(gè)類是一個(gè)Cloner類,其有一個(gè)Content類型的成員,然后有個(gè)構(gòu)造函數(shù)可以初始化成員,最后有一個(gè)GetCopy的方法,通過MemberwiseClone方法來復(fù)制自己。
下面我們通過一段代碼來調(diào)用Cloner類:
復(fù)制代碼 代碼如下:
static void Main(string[] args)
{
Cloner source = new Cloner(10);
Cloner target = (Cloner)source.GetCopy();//返回的是一個(gè)Object類型 需要做類型轉(zhuǎn)換.
Console.WriteLine("target.MyContent.Val = {0}", target.MyContent.Val);
source.MyContent.Val = 15;
Console.WriteLine("target.MyContent.Val = {0}", target.MyContent.Val);
Console.ReadKey();
}
結(jié)果是:
![](http://img.jbzj.com/file_images/article/201411/20141117102825500.png?2014101710297)
我們可以看到我們通過GetCopy()函數(shù)復(fù)制類source給target,但是當(dāng)我們改變source的時(shí)候,source輸出的值也跟著改變。由此我們可以得出,我們通過MemberwiseClone()復(fù)制的只是引用,即source和target的MyContent是相同的對(duì)象實(shí)例。這就是淺度復(fù)制,那我們?nèi)绾螌?shí)現(xiàn)深度復(fù)制呢,在.NET Framework中,給我們提供了ICloneable接口。
首先我們看下ICloneable接口:
復(fù)制代碼 代碼如下:
// 摘要:
// 支持克隆,即用與現(xiàn)有實(shí)例相同的值創(chuàng)建類的新實(shí)例。
[ComVisible(true)]
public interface ICloneable
{
// 摘要:
// 創(chuàng)建作為當(dāng)前實(shí)例副本的新對(duì)象。
//
// 返回結(jié)果:
// 作為此實(shí)例副本的新對(duì)象。
object Clone();
}
在上面那個(gè)例子中,我們只需要修改一些代碼就可以了:
復(fù)制代碼 代碼如下:
public class Cloner:ICloneable
{
public Content MyContent = new Content();
public Cloner(int newVal)
{
MyContent.Val = newVal;
}
//public object GetCopy()
//{
// return MemberwiseClone();
//}
public object Clone()
{
Cloner cloned = new Cloner(MyContent.Val);
return cloned;
}
}
為了做區(qū)別我把之前的代碼放在上面,注釋了的代碼就是淺度復(fù)制,后面的為深度復(fù)制,我們看到差別在于用本實(shí)例的MyContent.Val重新生成了實(shí)例返回給目標(biāo),測試結(jié)果為:
![](http://img.jbzj.com/file_images/article/201411/20141117102832953.png?20141017102856)
這里如果MyContent的成員不是一個(gè)值類型,那么我們還需要進(jìn)行深度,像下面一樣:
復(fù)制代碼 代碼如下:
public object Clone()
{
Cloner cloned = new Cloner();
cloned.MyContent = MyContent.Clone();
return cloned;
}
深度復(fù)制和淺度復(fù)制就是這樣,關(guān)鍵是創(chuàng)建一個(gè)新的對(duì)象實(shí)例返回去,而不是把原來的對(duì)象實(shí)例返回回去。
希望本文所述對(duì)大家的.NET程序設(shè)計(jì)有所幫助。
您可能感興趣的文章:- C# 對(duì)文件與文件夾的操作包括刪除、移動(dòng)與復(fù)制
- C# Bitmap 復(fù)制的小例子
- c#根據(jù)文件大小顯示文件復(fù)制進(jìn)度條實(shí)例
- C#使用SqlBulkCopy批量復(fù)制數(shù)據(jù)到數(shù)據(jù)表
- C#進(jìn)行文件讀寫、創(chuàng)建、復(fù)制、移動(dòng)、刪除的方法
- C#編程自學(xué)之?dāng)?shù)據(jù)類型和變量一
- C#編程自學(xué)之?dāng)?shù)據(jù)類型和變量二
- C#編程自學(xué)之?dāng)?shù)據(jù)類型和變量三
- C#裝箱和拆箱原理詳解
- C#編程自學(xué)之運(yùn)算符和表達(dá)式
- C#中深度復(fù)制和淺度復(fù)制詳解