c# 如何深拷贝和浅拷贝
技术百科
幻夢星雲
发布时间:2026-01-02
浏览: 次 浅拷贝用MemberwiseClone()仅复制第一层引用,新旧对象共享引用类型子对象;深拷贝推荐System.Text.Json序列化反序列化,安全高效;Newtonsoft.Json更灵活但需引入第三方包;禁用BinaryFormatter。
浅拷贝用 MemberwiseClone(),但只复制第一层引用
在 C# 中,MemberwiseClone() 是最直接的浅拷贝方式,它会创建新对象,并把原对象所有字段值(包括引用)逐字节复制过去。对值类型字段是真正复制,对引用类型字段只是复制“引用地址”,所以新旧对象仍共享同一堆内存中的子对象。
常见错误现象:修改拷贝后对象里的 List 或自定义类实例,原对象也跟着变——这就是浅拷贝的典型副作用。
使用场景有限:仅适用于字段全是值类型、或你明确知道引用字段不需要独立副本的情况(比如只读缓存对象)。
- 必须是
class,且不能是sealed以外的限制(但MemberwiseClone()本身是protected,需在类内部调用) - 不支持跨程序集深拷贝,也不处理循环引用
- 性能高,但语义风险大,生产环境慎用
深拷贝推荐用 System.Text.Json 序列化反序列化
对大多数 POCO 类型,用 System.Text.Json 是目前最轻量、安全、无需额外依赖的深拷贝方案。它把对象转成 JSON 字符串再解析回来,天然切断所有引用关系。
示例:
var original = new Person { Name = "Alice", Address = new Address { City = "Beijing" } };
var clone = JsonSerializer.Deserialize(
JsonSerializer.Serialize(original)); 注意点:
- 目标类型必须有无参构造函数(
System.Text.Json默认需要) - 不支持
DateTimeOffset等部分类型默认序列化(需配置JsonSerializerOptions) - 忽略
[JsonIgnore]或[XmlIgnore]属性,但不会跳过private字段(默认行为) - 性能比
BinaryFormatter好,且无反序列化远程代码执行风险
需要保
留类型/字段访问控制时用 Newtonsoft.Json
当你的类有 private set、readonly 字段,或继承结构复杂(如抽象基类 + 多个派生类),System.Text.Json 可能无法还原原始字段状态,这时 Newtonsoft.Json 更灵活。
它支持 JsonPropertyAttribute 控制序列化粒度,也能通过 PreserveReferencesHandling.Objects 处理循环引用(虽然深拷贝一般不希望保留引用)。
关键配置示例:
var settings = new JsonSerializerSettings
{
ConstructorHandling = ConstructorHandling.AllowNonPublicDefaultConstructor,
ReferenceLoopHandling = ReferenceLoopHandling.Ignore
};
var clone = JsonConvert.DeserializeObject(
JsonConvert.SerializeObject(original, settings), settings); 缺点也很明显:
- 引入第三方 NuGet 包(
Newtonsoft.Json) - 序列化性能略低于
System.Text.Json - 若未设
ReferenceLoopHandling,遇到循环引用直接抛JsonSerializationException
避免踩坑:别用 BinaryFormatter 做深拷贝
BinaryFormatter 曾是 .NET Framework 时代常用的深拷贝手段,但它在 .NET 5+ 已被标记为 不安全且废弃,官方明确不建议用于任何新代码。
原因很实在:
- 反序列化过程可能执行任意代码(CVE-2017-0199 类风险)
- 不支持跨版本兼容(.NET Core 和 .NET Framework 的二进制格式不一致)
- 要求所有类型都标注
[Serializable],且字段不能含不可序列化的资源(如文件句柄、数据库连接)
如果你在老项目里看到 BinaryFormatter.Deserialize() 做拷贝,优先替换为 System.Text.Json 方案,哪怕要补几个 [JsonInclude] 或调整构造函数。
真正难的不是选哪种方式,而是判断“这个对象到底需不需要深拷贝”——比如一个只读的 DTO 传参,根本不需要拷贝;而一个正在被多线程修改的配置对象,浅拷贝可能引发竞态。动手前先想清楚数据生命周期和所有权。
# 几个
# 也不
# 多个
# 不需要
# 第三方
# 更灵活
# 不支持
# js
# json
# 循环
# 对象
# 堆
# String
# class
# 值类型
# 字节
# c#
# 构造函数
# 字符串
# 序列化
# 数据库
# .net
# 线程
# private
# 多线程
# 继承
# 引用类型
# protected
# 第一层
# 句柄
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- php和redis连接超时怎么办_phpredis
- Linux怎么禁止Root用户远程登录_Linux
- 如何优化Golang Web性能_Golang H
- Win11怎么修复系统文件_使用sfc命令修复Wi
- c++怎么操作redis数据库_c++ hired
- Win10怎么卸载爱奇艺_Win10彻底卸载爱奇艺
- PythonDocker高级项目部署教程_多容器管
- 如何使用Golang指针与接口结合_实现方法调用和
- Windows11怎样开启游戏模式_Windows
- Windows系统被恶意软件破坏后的恢复策略_错误
- Win10系统怎么查看网络连接状态_Windows
- c++ unordered_map怎么用 c++哈
- Win11怎么压缩文件 Win11自带压缩解压功能
- 如何用正则表达式精确匹配最多含一个换行符的起止片段
- 如何使用Golang实现函数指针_函数变量与回调示
- Windows10怎么卸载预装软件_Windows
- Mac的Time Machine怎么用_Mac系统
- 如何使用Golang进行HTTP服务性能测试_测量
- Windows10系统怎么查看显卡驱动_Win10
- Python文件和流处理指南_高效读写大体积数据文
- Windows怎样关闭桌面弹窗广告_Windows
- Win11怎么清理C盘虚拟内存_Win11清理虚拟
- php文件怎么变mp4保存_php输出视频流保存为
- Win10电脑怎么设置网络名称_Windows10
- Linux如何安装JDK11_Linux环境变量配
- 如何在Golang中处理模块包路径变化_Golan
- 如何使用Golang实现微服务事件驱动_使用消息总
- Win11系统占用空间大怎么办 Win11深度瘦身
- Win11怎么调整屏幕亮度_Windows 11调
- Win11怎么设置虚拟内存_Windows 11优
- mac怎么分屏_MAC双屏显示与分屏操作技巧【指南
- php转mp4怎么设置帧率_调整php生成mp4视
- php会话怎么开启_session_start函数
- php在Linux怎么部署_LNMP环境搭建PHP
- PHP主流架构怎么部署到Docker_容器化流程【
- Win10如何卸载WindowsDefender_
- Mac如何开启夜览模式_Mac护眼模式设置与定时
- 如何在 ACF 中正确更新嵌套多层 Group 字
- Win11如何设置开机问候语 Win11修改登录界
- 短链接怎么用php递归还原_多层加密链接的处理法【
- VSC里PHP变量未定义报错怎么解决_错误抑制技巧
- Win11怎么检查TPM2.0模块_Windows
- Win10文件历史记录怎么用 Win10开启自动备
- 如何开启Windows的远程服务器管理工具(RSA
- Win11任务栏天气怎么关闭 Win11隐藏天气小
- 短链接怎么自定义还原php_修改解码规则适配需求【
- Win11局域网共享怎么设置 Win11文件夹网络
- Win11怎么开启自动HDR画质_Windows1
- Windows 11怎么更改锁屏超时时间_Wind
- Win10如何更改用户账户控制_Windows10

留类型/字段访问控制时用
QQ客服