热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C#编程指南:ShouldSerialize方法与特定条件下的序列化策略解析

ShouldSerialize*()vs*指定的条件序列化模式我知道ShouldSerialize*模式和*Specified模式以及它们是如何工作的,但两者之间有什么区别吗?当某

ShouldSerialize *()vs *指定的条件序列化模式

我知道ShouldSerialize *模式和* Specified模式以及它们是如何工作的,但两者之间有什么区别吗?

当某些事物应该有条件地序列化时,是否有任何“陷阱”使用一种方法而不是另一种方法?

此问题特定于XmlSerializer的用法,但也欢迎有关此主题的一般信息。

关于这个主题的信息非常少,所以可能是因为它们执行完全相同的目的而且它是一种风格选择。 但是,.NET实现者通过reflection分析类并查找其中一个/两个模式来确定生成的序列化程序的行为,因为它减慢了序列化程序的生成,除非它只是一个向后兼容性工件,这似乎很奇怪。

编辑:对于那些不熟悉这两种模式的人,如果*Specified属性或ShouldSerialize*方法返回true,则该属性被序列化。

 public string MyProperty { get; set; } //*Specified Pattern [XmlIgnore] public bool MyPropertySpecified { get{ return !string.IsNullOrWhiteSpace(this.MyProperty); } } //ShouldSerialize* Pattern public bool ShouldSerializeMyProperty() { return !string.IsNullOrWhiteSpace(this.MyProperty); } 

{propertyName}Specified模式的目的记录在XML Schema Binding Support:MinOccurs Attribute Binding Support中 。 添加它是为了支持XSD架构元素,其中:

在这种情况下, xsd.exe /classes将自动生成(或者您可以手动生成)与schema元素同名的属性和{propertyName}Specified boolean get / set属性,该属性用于跟踪XML中是否遇到该元素并应该序列化回XML。 如果遇到该元素, {propertyName}Specified设置为true ,否则设置为false 。 因此,反序列化的实例可以确定该属性是否在原始XML中未设置(而不是显式设置为其默认值)。

对于模式生成也实现了逆。 如果使用与上述模式匹配的一对属性定义C#类型,则使用xsd.exe生成相应的XSD文件,将相应的minOccurrs添加到模式中。 例如,给定以下类型:

 public class ExampleClass { [XmlElement] public decimal Something { get; set; } [XmlIgnore] public bool SomethingSpecified { get; set; } } 

将生成以下架构,反之亦然:

         

请注意,虽然xsd.exe仅记录为自动为值类型属性生成{propertyName}Specified属性,但XmlSerializer在手动使用引用类型属性时将遵循该模式。

你可能会问,为什么在这种情况下xsd.exe没有绑定到Nullable ? 也许是因为:

您需要了解此模式,因为xsd.exe有时会自动为您生成它,但是属性与其Specified属性之间的交互很奇怪并且容易产生错误。 您可以填写类中的所有属性,然后序列化为XML并丢失所有内容,因为您还没有将相应的Specified属性设置为true 。 这个“问题”不时出现在这里,例如看这个问题或者这个 问题 。

使用此模式的另一个“问题”是,如果您需要使用不支持此模式的序列化程序序列化您的类型,您可能希望在序列化期间手动抑制此属性的输出,并且可能需要在反序列化期间手动设置它。 由于每个序列化程序可能都有自己的自定义机制来抑制属性(或根本没有机制!),这样做会随着时间的推移变得越来越繁重。

(最后,我有点惊讶你的MyPropertySpecified没有setter的情况下成功运行。我似乎回想起.Net 2.0的一个版本,其中缺少{propertyName}Specified setter会引发exception。但它不再可重现在以后的版本中,我没有2.0测试。所以这可能是第三个问题。)

Windows窗体控件中的属性中记录了对ShouldSerialize{PropertyName}()方法的支持:使用ShouldSerialize和Reset方法定义默认值 。 正如您所看到的,文档位于MSDN的Windows窗体部分而不是XmlSerializer部分,因此它实际上是半隐藏function。 我不知道为什么在XmlSerializer存在对此方法和Specified属性的支持。 ShouldSerialize是在.Net 1.1中引入的,我相信在.Net 2.0中添加了MinOccurs绑定支持,所以早期的function可能不能完全满足xsd.exe开发团队的需求(或品味)?

因为它是一种方法而不是属性,所以它缺少{propertyName}Specified模式的“陷阱”。 它在实践中似乎也更受欢迎,并已被其他序列化程序采用,包括:

那么,使用哪种模式?

  1. 如果xsd.exe自动为您生成{propertyName}Specified属性,或者您的类型需要跟踪XML文件中是否出现特定元素,或者您需要自动生成的XSD来指示某个值是可选的,使用这种模式并注意“陷阱”。

  2. 否则,使用ShouldSerialize{PropertyName}()模式。 它的陷阱较少,可能会得到更广泛的支持。

为了添加@dbc的详细答案,我遇到了在派生类中管理序列化的问题。 在我的情况下,我有一个基类和一个派生类,其中一个Prop属性被覆盖。

 public class BaseClass { public virtual string Prop {get; set;} } public class Derived: BaseClass { public string Comp1 {get; set;} public string Comp2 {get; set;} public override string Prop {get => Comp1 + Comp2; set {}} } 

由于计算了Derived类中的Prop属性,因此对于Derived类,我想序列化Comp1Comp2而不是Prop 。 事实certificate,在Derived类中的Prop属性上设置XmlIgnore属性不起作用,并且Prop无论如何都是序列化的。

我还尝试在Derived类中添加ShouldSerializeProp方法和PropSpecified属性,但都不起作用。 我尝试设置断点以查看它们是否被调用而它们不是。

事实certificate, XmlSerializer正在查看原始类,其中Prop属性首次出现在类层次结构中,以决定是否序列化属性。 为了能够控制派生类中的序列化,首先我必须在Base类中添加一个virtual ShouldSerializeProp

 public class Base { ..... public virtual bool ShouldSerializeProp() {return true;} } 

然后我可以覆盖Derived类中的ShouldSerializeProp并返回false。

 public class Derived: Base { ..... public override bool ShouldSerializeProp() {return false;} } 

此模式允许不同的派生类从它们序列化的父类中选择哪些属性。 希望这可以帮助。

上述就是C#学习教程:ShouldSerialize *()vs *指定的条件序列化模式分享的全部内容,如果对大家有所用处且需要了解更多关于C#学习教程,希望大家多多关注—编程笔记


推荐阅读
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • PyCharm下载与安装指南
    本文详细介绍如何从官方渠道下载并安装PyCharm集成开发环境(IDE),涵盖Windows、macOS和Linux系统,同时提供详细的安装步骤及配置建议。 ... [详细]
  • CentOS7源码编译安装MySQL5.6
    2019独角兽企业重金招聘Python工程师标准一、先在cmake官网下个最新的cmake源码包cmake官网:https:www.cmake.org如此时最新 ... [详细]
  • c# – UWP:BrightnessOverride StartOverride逻辑 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
  • 深入理解OAuth认证机制
    本文介绍了OAuth认证协议的核心概念及其工作原理。OAuth是一种开放标准,旨在为第三方应用提供安全的用户资源访问授权,同时确保用户的账户信息(如用户名和密码)不会暴露给第三方。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 深入理解 SQL 视图、存储过程与事务
    本文详细介绍了SQL中的视图、存储过程和事务的概念及应用。视图为用户提供了一种灵活的数据查询方式,存储过程则封装了复杂的SQL逻辑,而事务确保了数据库操作的完整性和一致性。 ... [详细]
  • 本文介绍了如何使用JQuery实现省市二级联动和表单验证。首先,通过change事件监听用户选择的省份,并动态加载对应的城市列表。其次,详细讲解了使用Validation插件进行表单验证的方法,包括内置规则、自定义规则及实时验证功能。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • QUIC协议:快速UDP互联网连接
    QUIC(Quick UDP Internet Connections)是谷歌开发的一种旨在提高网络性能和安全性的传输层协议。它基于UDP,并结合了TLS级别的安全性,提供了更高效、更可靠的互联网通信方式。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文探讨了领域驱动设计(DDD)的核心概念、应用场景及其实现方式,详细介绍了其在企业级软件开发中的优势和挑战。通过对比事务脚本与领域模型,展示了DDD如何提升系统的可维护性和扩展性。 ... [详细]
  • 本文探讨了dbforms框架的核心设计理念及其背后的技术原理,详细分析了该框架如何通过其独特的设计模式来简化开发流程,并为开发者提供了优化使用方法的建议。 ... [详细]
  • Struts与Spring框架的集成指南
    本文详细介绍了如何将Struts和Spring两个流行的Java Web开发框架进行整合,涵盖从环境配置到代码实现的具体步骤。 ... [详细]
author-avatar
秋知落叶冷
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有