我今天注意到Newtonsoft.Json的一些奇怪的输出,我不确定它是否与F#类型的交互或C#中可能发生的事情,所以我已经标记了两者。我有一个序列化的以下记录列表:
var optionText, array = [], newString, maxChar = 40;
$('select').each(function(){
$(this).find('option').each(function(i,e) {
$(e).attr('title',$(e).text());
optionText = $(e).text();
newString = '';
if (optionText.length > maxChar) {
array = optionText.split(' ');
$.each(array,function(ind,ele) {
newString += ele+' ';
if (ind > 0 && newString.length > maxChar) {
$(e).text(newString);
return false;
}
});
}
});
});
我用 type SplitTracker =
{
[<JsonIgnore>]
split : SplitDefinition
mutable start : duration
mutable ``end`` : duration
mutable lapCount : int
mutable duration : duration Option
}
序列化它,我得到以下奇数输出:
JsonConvert.SerializeObject
任何人都知道为什么会发生这种情况?数据是正确的,带有&#34; @&#34;的字段重复。符号就是问题。
答案 0 :(得分:6)
您定义记录的方式是罪魁祸首。记录字段作为属性公开 - 但您使用的是可变属性。 F#会把它变成一个类,它包含每个mutable的字段(名称是属性名称,前缀为@),以及读出这些字段的属性。
Json现在将尝试序列化所有字段和所有属性 - 因此您可以获得重复。
在F#interactive中尝试:
type SplitTracker =
{
mutable start : float
}
let t = typeof<SplitTracker>
let fields1 = t.GetFields() // This will give you a field '@start'
let props1 = t.GetProperties() // This will give you a property 'start'
与使用普通记录时得到的对比:
type SplitTracker2 =
{
start : float
}
let t2 = typeof<SplitTracker2>
let fields2 = t2.GetFields() // You will not see any fields
let props2 = t2.GetProperties() // There is a single property 'start'
这应该正确序列化。除此之外,它使你的代码更加惯用。