C#深浅拷贝的深入解析
作者:小世界的野孩子 发布时间:2023-03-28 18:36:28
前言
前面我们学习完了设计模式,在其中我们有了解到原型模式。这里涉及到了克隆自身对象。那么也就是对对象进行拷贝。这里就涉及到了这么一个概念。深浅拷贝、何为深拷贝何为浅拷贝呢?我们一起来看看吧。
浅拷贝
首先我们看看浅拷贝。浅拷贝就是将对象中的所有字段复制到新对象中去,浅拷贝对于值类型和引用类型有不同的影响。值类型的值被复制到副本中后,修改副本中的值不会影响原来对象的值。然而引用类型被复制到副本中的是引用类型的引用。不是引用的对象。这样再修改副本中的值是会导致原来对象的值也被修改了。但是这里引用类型情况我们需要排除字符串String类型。
那么为何引用类型修改副本的值会造成原来对象的值的变化,而string字符串类型却排除在外呢?首先我们需要知道这么一个概念,string类型是一个不可变的数据类型,也就是意味着对字符串对象进行了初始化,该字符串对象就不能改变了。表面上我们修改字符串的内容的方法和运算实际上是创建了一个新字符串,然后根据需要可以把旧字符串的内容复制到新字符串中。怎么理解你?我们看下面这个案例:
#region 字符串比较
/// <summary>
/// 获取引用类型的内存地址方法
/// </summary>
/// <param name="o"></param>
/// <returns></returns>
public static string getMemory(object o)
{
GCHandle h = GCHandle.Alloc(o, GCHandleType.Pinned);
IntPtr addr = h.AddrOfPinnedObject();
return "0x" + addr.ToString("X");
}
/// <summary>
/// 字符串比较
/// </summary>
public static void Compares()
{
string a = "123";
Console.WriteLine("a的引用地址:\t\t" + getMemory(a));
string b = "123";
Console.WriteLine("b的引用地址:\t\t" + getMemory(b));
Console.WriteLine("a与b的比较:\t\t" + Object.ReferenceEquals(a, b));
b = "456";
Console.WriteLine("b的引用地址:\t\t" + getMemory(b));
}
#endregion
这里我们看a=”123”,b=”123”。我们看他们的引用地址是一样的。也就是说我们先创建a的时候创建了字符串a,有了一个引用地址。然后我们创建b的时候首先会寻找是否存在相同的值。如果存在相同的值就获取其引用地址。这也就是为什么a与b的引用地址是一样的。这里涉及到一个叫做字符驻留池的东西。会对字符串进行保存。那么后面我们修改b的值然后输出其引用地址,发现和之前的引用地址不一样。说明并不是修改原来的值,而是重新创建了一个字符串,重新获取了它的引用地址。
我们接下来看一个浅拷贝的案例吧,首先我们准备的是以下的数据类型的值:int,string,enum,struct,class,int[],string[]。
/// <summary>
/// 枚举
/// </summary>
public enum EnumTest
{
TestOne = 1,
TestTwo = 2
}
/// <summary>
/// 结构体
/// </summary>
public struct StructTest
{
public int Test;
public StructTest(int i)
{
Test = i;
}
}
/// <summary>
/// 类
/// </summary>
public class ClassTest
{
public string TestString;
public ClassTest(string _string)
{
TestString = _string;
}
}
/// <summary>
/// 深拷贝
/// </summary>
public class DeepClone : ICloneable
{
public int _int = 1;
public string _string = "1";
public EnumTest _enum = EnumTest.TestOne;
public StructTest _struct = new StructTest(1);
public ClassTest _class = new ClassTest("1");
public int[] arrInt = new int[] { 1 };
public string[] arrString = new string[] { "1" };
public object Clone()
{
var NewOne = JsonConvert.SerializeObject(this);
return JsonConvert.DeserializeObject<DeepClone>(NewOne);
}
}
class Program
{
static void Main(string[] args)
{
DeepClone simple = new DeepClone();
var simpleTwo = (DeepClone)simple.Clone();
simpleTwo._int = 2;
simpleTwo._string = "2";
simpleTwo._enum = EnumTest.TestTwo;
simpleTwo._struct.Test = 2;
simpleTwo._class.TestString = "2";
simpleTwo.arrInt[0] = 2;
simpleTwo.arrString[0] = "2";
Console.WriteLine($"int 类型变化 原对象:{simple._int}\t\t 备份对象:{simpleTwo._int}");
Console.WriteLine($"string 类型变化 原对象:{simple._string}\t\t 备份对象:{simpleTwo._string}");
Console.WriteLine($"enum 类型变化 原对象:{(int)simple._enum}\t\t 备份对象:{(int)simpleTwo._enum}");
Console.WriteLine($"struct 类型变化 原对象:{simple._struct.Test}\t\t 备份对象:{simpleTwo._struct.Test}");
Console.WriteLine($"class 类型变化 原对象:{simple._class.TestString}\t\t 备份对象:{simpleTwo._class.TestString}");
Console.WriteLine($"int数组 类型变化 原对象:{simple.arrInt[0]}\t\t 备份对象:{simpleTwo.arrInt[0]}");
Console.WriteLine($"string数组 类型变化 原对象:{simple.arrString[0]}\t\t 备份对象:{simpleTwo.arrString[0]}");
}
}
我们通过继承ICloneable接口对这些类型都进行了浅拷贝然后修改副本对象。输出原对象和副本对象进行比较。我们发现int,enum,struct、值类型以及string这个特殊的引用类型的原对象值没有被影响改变。但是class,int[],string[]这些引用类型对象原对象被影响改变了值。也就再次验证了我们前面说的。浅拷贝是将对象进行赋值到一个副本对象中去,值类型复制值,引用类型复制其引用对象。修改副本对象值,值类型和string原对象不会被影响改变,引用类型除string其原对象都会被影响改变。
深拷贝
我们上面看了浅拷贝,浅拷贝还是有一定的影响的,处理不好可能就成bug。那么我们看看对应的深拷贝又是什么样的呢?这里可以先声明,深拷贝对值类型和引用类型都没有区别对待。深拷贝也是将对象中的所有字段复制到新对象中去,但是对象无论是值类型还是引用类型都将被重新创建然后复制到副本对象去。对于副本对象的修改将不会影响到原对象,无论任何类型。
我们继续将上面的例子进行深拷贝看看:
/// <summary>
/// 深拷贝
/// </summary>
public class DeepClone : ICloneable
{
public int _int = 1;
public string _string = "1";
public EnumTest _enum = EnumTest.TestOne;
public StructTest _struct = new StructTest(1);
public ClassTest _class = new ClassTest("1");
public int[] arrInt = new int[] { 1 };
public string[] arrString = new string[] { "1" };
public object Clone()
{
var NewOne = JsonConvert.SerializeObject(this);
return JsonConvert.DeserializeObject<DeepClone>(NewOne);
}
}
class Program
{
static void Main(string[] args)
{
DeepClone simple = new DeepClone();
var simpleTwo = (DeepClone)simple.Clone();
simpleTwo._int = 2;
simpleTwo._string = "2";
simpleTwo._enum = EnumTest.TestTwo;
simpleTwo._struct.Test = 2;
simpleTwo._class.TestString = "2";
simpleTwo.arrInt[0] = 2;
simpleTwo.arrString[0] = "2";
Console.WriteLine($"int 类型变化 原对象:{simple._int}\t\t 备份对象:{simpleTwo._int}");
Console.WriteLine($"string 类型变化 原对象:{simple._string}\t\t 备份对象:{simpleTwo._string}");
Console.WriteLine($"enum 类型变化 原对象:{(int)simple._enum}\t\t 备份对象:{(int)simpleTwo._enum}");
Console.WriteLine($"struct 类型变化 原对象:{simple._struct.Test}\t\t 备份对象:{simpleTwo._struct.Test}");
Console.WriteLine($"class 类型变化 原对象:{simple._class.TestString}\t\t 备份对象:{simpleTwo._class.TestString}");
Console.WriteLine($"int数组 类型变化 原对象:{simple.arrInt[0]}\t\t 备份对象:{simpleTwo.arrInt[0]}");
Console.WriteLine($"string数组 类型变化 原对象:{simple.arrString[0]}\t\t 备份对象:{simpleTwo.arrString[0]}");
}
}
这里我们看这个运行结果,无论值类型还是引用类型修改副本对象之后都没有影响原对象的值。这也就是深拷贝的特点了。
总结
我们看完了浅拷贝与深拷贝,我们仔细回顾下。浅拷贝将对象的字段复制到新的对象中去,但是当修改新对象的时候,值类型和string类型的字段将不会影响原对象的字段,而引用类型除string类型外都将影响原对象的值。深拷贝也是将对象的字段复制到新的对象中去,但是无论是值类型还是引用类型的改变都不会影响原对象的值。因为深拷贝是将原对象重新创建然后复制到副本对象中去的。
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。
来源:https://www.cnblogs.com/hulizhong/p/11776610.html


猜你喜欢
- 前言:本人目前从事java开发,但同时也在学习各种前端技术,下面是我做的一个前后端分离项目的一个小案例,不足之处请多多指教1. 项目技术选型
- 如何高效地进行数组复制?如果把一个变量值复制给另外一个数组变量,那么2个变量指向托管堆上同一个引用。如果想在托管堆上创建另外的一份数组实例,
- 函数式接口函数式接口(functional interface 也叫功能性接口,其实是同一个东西)。简单来说,函数式接口是只包含一个方法的接
- 老生常谈的配置 但是还是需要说明一下EurekaApplication @EnableEurekaServer指定为server端
- 背景随着公司业务越来越复杂,在同一个列表中需要展示各种类型的数据。总体结构ItemViewAdapter: 每种类型的卡片分别都是不同的It
- 前言当用户向服务器发送了一次HTTP请求,该请求可能会经过多个信息资源处理以后才返回给用户,各个信息资源使用请求转发机制相互转发请求,但是用
- 发现问题最近在进行压测发现,有一些接口时好时坏,通过sentry日志平台及sky walking平台跟踪发现,用户张三获取到的用户上下文确是
- 一、Jsoup的主要功能Jsoup是一款Java 的HTML解析器,可直接解析某个URL地址、HTML文本内容。它提供了一套非常省力的API
- 1.获取签名与模板进入阿里云平台,进入短信服务模块,在以下位置添加签名和模板(格式一定按照要求填写 审批的比较严格)2.编写模板与签名的枚举
- 1、什么是反射?在java开发中有一个非常重要的概念就是java反射机制,也是java的重要特征之一。反射的概念是由Smith在1982年首
- 近几年移动互联网的高速发展,智能手机的使用用户呈现 * 性增长,手机终端上的App 种类繁多,大多数App 都需要与后台系统进行交互,交互的第
- 实例如下:package com.ljq.util;import java.beans.BeanInfo;import java.beans
- Java 7的这个新特性改变了警告的对象。构建这些类型毕竟有破坏类型安全的风险,这总得有人知道。但 API 的用户对此是无能为力的,不管do
- 什么是FTPFTP(File Transfer Protocol)是TCP/IP网络上两台计算机传送文件的协议,使得主机间可以共享文件.可以
- 最近在做一个项目,遇到了项目打成 war 包的一个问题,项目创建时选择的时 jar 包方式,后因项目部署要求,需要打成 war 包部署,遇到
- controller传boolean形式值@GetMapping("/check-cart")public List&l
- 首先:我们要建一个web项目接着: 我们先来导入struts的xml文件第一步:右击你的项目名,鼠标到MyEclipse会看到一个add s
- java中javaBean与Bean的深入理解JavaBean 是Java中的一种特殊的类,可以将多个对象封装到一个对象(bean)中。特点
- 前言做Android的这两年时间,通过研究Android源码,也会Java并发处理多线程有了自己的一些理解。那么问题来了,如何实现一个串行的
- 效果图白话分析:多线程:肯定是多个线程咯断点:线程停止下载的位置续传:线程从停止下载的位置上继续下载,直到完成任务为止。核心分析:断点:当前