从JObject转换为XDocument时需要保留尾随零

时间:2013-07-29 21:00:26

标签: c# xml json json.net

我有一个相当简单的请求。我在C#中创建一个json对象。我有一些属性是双打的。最初,我创建了xml并将这些值格式化为在小数点后面有适当数量的尾随零。现在我正在创建json和xml。我使用LINQ to JSON生成json。然后我使用JSONConvert创建和XDocument。 但问题是,如果我将double值作为对象而不是格式化的字符串放入JObject中,我会丢失尾随零,而不是将其存储为字符串。
以下是一些示例代码:

double value = 1.0;
string stringVal = "1.000";
JObject test = new JObject(new JProperty("sta", new JObject(
        new JProperty("@valDouble", value),
        new JProperty("@valString", stringVal)
        )));
var testXml = JsonConvert.DeserializeXNode(test.ToString());

我意识到我可以通过我创建的JObject并提取值并使用代码手动将其格式化为字符串。但是有一些方法可以让转换器使用某种格式处理值。所以没有得到

<sta valDouble="1" valString="1.000">
</sta>

我想知道,

<sta valDouble="1.000" valString="1.000">
</sta>

我希望我已经很好地解释了这个问题。我已经查看了Deserialize方法,但是它们只是有其他选项来选择根元素名称以及是否编写数组属性。在这种情况下,他们没有帮助我。

1 个答案:

答案 0 :(得分:0)

好的,我实际上找到了一个解决方案,但它需要修改版本的XmlNodeConverter。为了保持我的影响很小,我实际上对实际代码进行了微小的更改,然后创建了一个派生类。

//added this virtual method
protected virtual string GetAttributeValue(string attributeName, JsonReader reader)
{
    return reader.Value.ToString();
}

private void ReadElement(JsonReader reader, IXmlDocument document, IXmlNode currentNode, string propertyName, XmlNamespaceManager manager)
{
  if (string.IsNullOrEmpty(propertyName))
    throw new JsonSerializationException("XmlNodeConverter cannot convert JSON with an empty property name to XML.");

  Dictionary<string, string> attributeNameValues = ReadAttributeElements(reader, manager);

  string elementPrefix = MiscellaneousUtils.GetPrefix(propertyName);

  if (propertyName.StartsWith("@"))
  {
    var attributeName = propertyName.Substring(1);
    //Made the change below to use the new method
    //var attributeValue = reader.Value.ToString();
    var attributeValue = GetAttributeValue(attributeName, reader);
    ...

// Modified to virtual to allow derived class to override.  Added attributeName parameter
protected virtual string ConvertTokenToXmlValue(string attributeName, JsonReader reader)
{
  if (reader.TokenType == JsonToken.String)
  {
    return reader.Value.ToString();
  }
  ....

然后我创建了以下派生的xml节点转换器。

public class DerivedXmlNodeConverter : Newtonsoft.Json.Converters.XmlNodeConverter
{
private Dictionary<string, string> _attributeFormatStrings;
public Dictionary<string, string> AttributeFormatStrings
{
    get
    {
        if (_attributeFormatStrings == null)
            _attributeFormatStrings = new Dictionary<string,string>();
        return _attributeFormatStrings;
    }
}
protected override string GetAttributeValue(string attributeName, JsonReader reader)
{
    Console.WriteLine("getting attribute value: " + attributeName);
    if (AttributeFormatStrings.ContainsKey(attributeName))
    {
        return string.Format(AttributeFormatStrings[attributeName], reader.Value);
    }
    else
        return base.GetAttributeValue(attributeName, reader);
}
protected override string ConvertTokenToXmlValue(string attributeName, JsonReader reader)
{
    if (AttributeFormatStrings.ContainsKey(attributeName))
    {
        return string.Format(AttributeFormatStrings[attributeName], reader.Value);
    }
    else
        return base.ConvertTokenToXmlValue(attributeName, reader);
} 
}

然后我不得不复制JsonConvert.DeserializeXNode所做的代码。

    DerivedXmlNodeConverter derived = new DerivedXmlNodeConverter();
    derived.WriteArrayAttribute = false;
    derived.DeserializeRootElementName = null;
    derived.AttributeFormatStrings["valDouble"] = "{0:0.000}";
    JsonSerializerSettings settings = new JsonSerializerSettings { Converters = new JsonConverter[] { derived } };
    StringReader sr = new StringReader(test.ToString(Newtonsoft.Json.Formatting.Indented));
    JsonReader reader = new JsonTextReader(sr);
    JsonSerializer ser = JsonSerializer.CreateDefault(settings);
    ser.CheckAdditionalContent = true;
    XDocument intoXml = (XDocument)(ser.Deserialize(reader, typeof(XDocument)));

我认为这些将是Newtownsoft框架中的重大变化 - 允许这些钩子进行自定义。 我希望这段代码可以帮助其他需要它的人。

-David