JsonConvert.SerializeObject - 自定义 ContracteResolver 不为多次调用调用 CreateProperty

Posted

技术标签:

【中文标题】JsonConvert.SerializeObject - 自定义 ContracteResolver 不为多次调用调用 CreateProperty【英文标题】:JsonConvert.SerializeObject - Custom ContracteResolver not calling CreateProperty for multiple calls 【发布时间】:2019-02-26 01:41:21 【问题描述】:

我有一个自定义 ContracteResolver,但我得到了不可预测的结果。

使用调试器,我看到当我序列化解析器的 CreateProperty 方法时,每个属性都会被调用。但是,如果我进行 2 次背靠背调用,则不会在第二次调用时调用 CreateProperty 方法。我在 CreateProperty 中的断点在第二遍时从未命中,而是在第一遍中命中。

这是我的设置:

IContractResolver contractResolver = new ShouldSerializeContractResolver(fieldsToSerialize, this.Data);
var settings = new JsonSerializerSettings()

ContractResolver = contractResolver
;

_payload = JsonConvert.SerializeObject(this.Data, Formatting.None, settings);

两个调用的源值 (this.Data) 不同。两个调用的结果(_payload)也不同。我认为没有任何内容被缓存。

我看到了一个由自定义 ContentNegotiator 引起的类似问题,但我没有使用它。

为什么 CreateProperty 不会在第二遍时被命中?

【问题讨论】:

ShouldSerializeContractResolver 继承自什么?是DefaultContractResolver 还是CamelCasePropertyNamesContractResolver 因为如果您从 CamelCasePropertyNamesContractResolver 继承,它会在应用程序域期间全局缓存所有合同信息。详情请见Json.Net: html Helper Method not regenerating。 【参考方案1】:

DefaultContractResolver 类缓存每种对象类型的协定以获得最佳性能。您可以在 source code 中看到这一点,它在构造函数中创建缓存:

public DefaultContractResolver()

   ...
    _contractCache = new ThreadSafeStore<Type, JsonContract>(CreateContract);

以及在为每种类型解析合同时如何使用它:

public virtual JsonContract ResolveContract(Type type)

    ValidationUtils.ArgumentNotNull(type, nameof(type));

    return _contractCache.Get(type);

如果:

    您的自定义解析器派生自DefaultContractResolver,并且 您每次都使用相同的解析器实例,并且 您的数据对象属于同一类型(即使它们具有不同的数据)

那么很正常,预计CreateProperty只会在第一次序列化时被调用。

【讨论】:

几乎是答案。我没有使用相同的解析器实例,但我从缓存在静态变量中的 CamelCasePropertyNamesContractResolver 继承。如果我从 DefaultContractResolver 继承它可以工作,但我丢失了我的 CamelCase。如何使用两个解析器?我正在根据数据添加和删除字段。 没有办法同时使用两个解析器。如果您只需要骆驼大小写行为并且您使用的是 Json.Net 9.0 或更高版本,则可以将默认解析器(或您的自定义衍生工具)上的 NamingStrategy 属性设置为 CamelCaseNamingStrategy 获取骆驼大小写命名。跨度> 我有同样的问题,但它根本没有发现它。

以上是关于JsonConvert.SerializeObject - 自定义 ContracteResolver 不为多次调用调用 CreateProperty的主要内容,如果未能解决你的问题,请参考以下文章