JsonConvert.serializeObject,数据集返回意外结果

时间:2017-11-20 12:59:10

标签: c# .net xml asp.net-core-2.0 .net-core-2.0

我正在尝试将数据集转换为.net core 2.0中的JSON,并使用以下代码执行此操作

return Ok(JsonConvert.SerializeObject(dataset, Formatting.Indented))

我得到的输出是

"{\n \"DataSet.RemotingVersion\": {\n \"Major\": 2,\n \"Minor\": 0,\n \"Build\": -1,\n \"Revision\": -1,\n \"MajorRevision\": -1,\n \"MinorRevision\": -1\n },\n \"XmlSchema\": \"<?xml version=\\\"1.0\\\" encoding=\\\"utf-16\\\"?>\\n<xs:schema id=\\\"dataset\\\" xmlns=\\\"\\\" xmlns:xs=\\\"http://www.w3.org/2001/XMLSchema\\\" xmlns:msdata=\\\"urn:schemas-microsoft-com:xml-msdata\\\">\\n <xs:element name=\\\"dataset\\\" msdata:IsDataSet=\\\"true\\\" msdata:UseCurrentLocale=\\\"true\\\">\\n <xs:complexType>\\n <xs:choice minOccurs=\\\"0\\\" maxOccurs=\\\"unbounded\\\">\\n <xs:element name=\\\"datatable\\\">\\n <xs:complexType>\\n <xs:sequence>\\n <xs:element name=\\\"cvbf\\\" type=\\\"xs:string\\\" msdata:targetNamespace=\\\"\\\" minOccurs=\\\"0\\\" />\\n </xs:sequence>\\n </xs:complexType>\\n </xs:element>\\n </xs:choice>\\n </xs:complexType>\\n </xs:element>\\n</xs:schema>\",\n \"XmlDiffGram\": \"<diffgr:diffgram xmlns:msdata=\\\"urn:schemas-microsoft-com:xml-msdata\\\" xmlns:diffgr=\\\"urn:schemas-microsoft-com:xml-diffgram-v1\\\" />\"\n}"

对于C#开发来说,这是一个新手,

  • 如何将其转换为适当的JSON?
  • 为什么会出现这样的情况?

2 个答案:

答案 0 :(得分:2)

这里有两个问题:

  1. 您似乎是按照JSON.NET Parser seems to be double serializing my objects中描述的方式对dataset进行了双重序列化。首先,您手动将返回的对象序列化为字符串,然后调用Ok(object) 序列化传入的对象(即JSON字符串),从而产生嵌套序列化。

    解决方案不是这样做的。让框架通过简单地执行return Ok(dataset);

  2. 为您序列化您的对象
  3. 你的第二个问题是你试图在.Net核心中使用Json.NET序列化DataSet - 不幸的是,目前还不支持这样做。正如Issue #1409: serializes a DataSet to JSON on dotnet core 2.0Issue #1383: .Net core build doesn't include DataTableConverter中所述,当前版本的Json.NET 10.0.3针对的是netstandard 1.3,而DataSetDataTable是在netstandard 1.5中引入的。因此,当前的.Net核心版本的Json.NET没有内置逻辑来序列化这些类型。您需要等到Json.NET的下一个版本,它将以netstandard 2.0为目标。

    那么,使用当前版本有哪些选择?首先,Json.NET在MIT License下获得许可,因此您可以复制其DataSetConverterDataTableConverter的版本,将它们包含在您的项目中,并将它们添加到JsonSerializerSettings.Converters。问题JsonSerializerSettings and Asp.Net Core显示了如何在ASP.NET Core中自定义设置。

    其次,您可以将DataSet转换为DTO并将其返回。例如,以下扩展方法将任何DataSet转换为Dictionary<string, List<Dictionary<string, object>>>,这可由Json.NET序列化:

    public static class DataSetExtensions
    {
        public static List<Dictionary<string, object>> ToDictionaryList(this DataTable table)
        {
            if (table == null)
                return null;
            return table.Rows.Cast<DataRow>()
                .Select(r => table.Columns.Cast<DataColumn>().ToDictionary(c => c.ColumnName, c => r[c]))
                .ToList();
        }
    
        public static Dictionary<string, List<Dictionary<string, object>>> ToDictionary(this DataSet set)
        {
            if (set == null)
                return null;
            return set.Tables.Cast<DataTable>()
                .ToDictionary(t => t.TableName, t => t.ToDictionaryList());
        }
    }
    

    更好的是,创建一个强类型模型来返回。 (如果您还要序列化为XML,那么这是更好的解决方案,因为XmlSerializer不支持字典序列化。)

答案 1 :(得分:1)

截至25/04/2018。我有完全相同的问题。下载最新版本的Newtonsoft。我升级到11.0.2。它现在适用于ASP Core 2.数据集转换为JSON。