[英]JSON.NET to C# objects
我试图在Windows窗体中使用JSON.NET框架从JSON字符串中读取一些信息。 但我努力从“分类法 - >主题”阵列和“集群”中获取字典
{
"keywords": {
"anyString": [
],
"allString": {
"a5349f533e3aa3ccbc27de2638da38d6": "olympics"
},
"exactString": [
],
"notString": [
],
"highlightString": [
]
},
"dates": {
"startDate": "15-01-2008",
"endDate": "15-09-2009",
"useDates": true
},
"clusters": {
"permission": {
"1": "private\/n"
}
},
"taxonomies": {
"Topics": {
"2488": "Olympics 2012 (not participation)",
"8876": "Olympics and culture"
},
"Keywords": {
"8848": "Engineering in the Olympics"
}
},
"sort": {
"sortId": 1,
"sortType": 2,
"sort": "datetime",
"sortOrder": "descending"
}
}
使用下面的代码,我已经能够阅读一些信息。
JObject searchCriteria = JObject.Parse(contentSearchCriteria);
//search criteria
IEnumerable<string> allString = searchCriteria["keywords"]["allString"].Children().Values<string>();
IEnumerable<string> anyString = searchCriteria["keywords"]["anyString"].Children().Values<string>();
IEnumerable<string> notString = searchCriteria["keywords"]["notString"].Children().Values<string>();
IEnumerable<string> exactString = searchCriteria["keywords"]["exactString"].Children().Values<string>();
IEnumerable<string> highlightString = searchCriteria["keywords"]["highlightString"].Children().Values<string>();
//dates
string startDate = (string)searchCriteria["dates"]["startDate"];
string endDate = (string)searchCriteria["dates"]["endDate"];
bool useDates = (bool)searchCriteria["dates"]["useDates"];
//sort
int sortId = (int)searchCriteria["sort"]["sortId"];
int sortType = (int)searchCriteria["sort"]["sortType"];
string sort = (string)searchCriteria["sort"]["sort"];
string sortOrder = (string)searchCriteria["sort"]["sortOrder"];
更新:
正如我所建议的那样
class SMSearchCriteria
{
public SMKeywords keywords { get; set; }
public SMDates dates { get; set; }
public SMClusters clusters { get; set; }
public SMTaxonomies taxonomies { get; set; }
public SMSort sort { get; set; }
}
class SMKeywords
{
public List<Dictionary<string, string>> AnyString {get; set;}
public List<Dictionary<string, string>> AllString { get; set; }
public List<Dictionary<string, string>> ExactString { get; set; }
public List<Dictionary<string, string>> NotString { get; set; }
public List<Dictionary<string, string>> HighlightString { get; set; }
}
class SMDates
{
public string startDate { get; set; }
public string endDate { get; set; }
public bool useDates { get; set; }
}
class SMClusters
{
List<SMCluster> cluster;
}
class SMCluster
{
public Dictionary<string, string> cluster { get; set; }
}
class SMTaxonomies
{
public List<SMTaxonomy> taxonomies { get; set; }
}
class SMTaxonomy
{
public Dictionary<string, List<SMCategory>> taxonomy { get; set; }
}
class SMCategory
{
public Dictionary<int, string> category { get; set; }
}
class SMSort
{
public int sortId { get; set; }
public int sortType { get; set; }
public string sort { get; set; }
public string sortOrder { get; set; }
}
但是当我执行时:
var mydata = JsonConvert.DeserializeObject<SMSearchCriteria>(contentSearchCriteria);
我得到了例外:
[Newtonsoft.Json.JsonSerializationException] = {"Cannot deserialize JSON object into type 'System.Collections.Generic.List`1[System.Collections.Generic.Dictionary`2[System.String,System.String]]'."}
更新2:
正如所建议的那样,我删除了所有额外的列表,并将类简化为:
class SearchMasterSearchCriteria
{
public SMKeywords keywords { get; set; }
public SMDates dates { get; set; }
public Dictionary<string, Dictionary<int, string>> clusters { get; set; }
public Dictionary<string, Dictionary<int, string>> taxonomies { get; set; }
public SMSort sort { get; set; }
}
class SMKeywords
{
public Dictionary<string, string> anyString {get; set;}
public Dictionary<string, string> allString { get; set; }
public Dictionary<string, string> exactString { get; set; }
public Dictionary<string, string> notString { get; set; }
public Dictionary<string, string> highlightString { get; set; }
}
class SMDates
{
public string startDate { get; set; }
public string endDate { get; set; }
public bool useDates { get; set; }
}
class SMSort
{
public int sortId { get; set; }
public int sortType { get; set; }
public string sort { get; set; }
public string sortOrder { get; set; }
}
我还添加了测试代码来序列化对象,如下所示:
//criteria
SearchMasterSearchCriteria smCriteria = new SearchMasterSearchCriteria();
//keywords
SMKeywords smKeywords = new SMKeywords(); ;
Dictionary<string, string> dict = new Dictionary<string, string>();
dict.Add("a5349f533e3aa3ccbc27de2638da38d6", "olympics");
dict.Add("9cfa7aefcc61936b70aaec6729329eda", "games");
smKeywords.allString = dict;
//category
Dictionary<int, string> categorieDict = new Dictionary<int, string>();
categorieDict.Add(2488, "Olympics 2012 (not participation)");
categorieDict.Add(8876, "Olympics and culture");
//taxonomies
Dictionary<string, Dictionary<int, string>> taxonomiesDict = new Dictionary<string, Dictionary<int, string>>();
taxonomiesDict.Add("Topics", categorieDict);
//metadata
Dictionary<int, string> metadataDict = new Dictionary<int, string>();
metadataDict.Add(1, @"private/n");
//clusters
Dictionary<string, Dictionary<int, string>> clustersDict = new Dictionary<string, Dictionary<int, string>>();
clustersDict.Add("permission", metadataDict);
//dates
SMDates smDates = new SMDates();
smDates.startDate = "15-01-2008";
smDates.endDate = "15-09-2009";
smDates.useDates = true;
//sort
SMSort smSort = new SMSort();
smSort.sortId = 1;
smSort.sortType = 2;
smSort.sort = "datetime";
smSort.sortOrder = "descending";
//add to criteria.
smCriteria.keywords = smKeywords;
smCriteria.clusters = clustersDict;
smCriteria.taxonomies = taxonomiesDict;
smCriteria.dates = smDates;
smCriteria.sort = smSort;
//serialize
string json = JsonConvert.SerializeObject(smCriteria);
var mydata1 = JsonConvert.DeserializeObject<SearchMasterSearchCriteria>(json);
到那个时候2 json字符串之间的唯一区别在哪里。 anyString,exactString等的[]和nulls所以我用卷曲的方括号替换空方括号,并且没有错误地去除了它们:)
contentSearchCriteria = contentSearchCriteria.Replace("[]", "{}");
var mydata = JsonConvert.DeserializeObject<SearchMasterSearchCriteria>(contentSearchCriteria);
说实话,我不会这样做,你完全是这样做的。 这是我检索数据的方式:
class Data {
Dictionary<string, Dictionary<string, string>> keywords;
DatesClass dates;
.......
}
class DatesClass
{
string startDate;
string endDate;
bool? useDates
}
var mydata = JsonConvert.DeserializeObject<Data>(jsonstring);
我没有填写整个数据类,但你明白了。 我发现在输入数据的结构中创建一个对象然后使用DeserializeObject方法来填充数据要容易得多。 这也使代码更清晰,并允许编译器检查拼写错误。
是的,你现在的问题是JSON.net反序列化对象的方式。 在JSON.net中,C#类成为Json对象。 并且该类的成员成为Key,成员的值变为值。
我们以分类法路径为例。 使用上面的类定义,JSON.net正在以这种格式查找Json数据:
{"taxonomies": {"taxonomies":[{"taxonomy": {"Topics": {1212, "foo"}}}]}
这看起来像你的输入数据一样。
当你创建对象时,请以这种方式思考它。
1)基础对象在json代码中创建{}。 2)字典在json代码中创建{}。 3)List在json代码中创建一个[] 4)类的每个成员在json代码的{}中创建一个条目
可能有助于为您调试这个是创建您的结构,填写一些临时数据然后使用JsonConvert.Serialize(myobj)向您展示JSON认为结构将是什么样子。
我认为你的例外来自于许多课程。
这可能是您希望代码的taxominies部分看起来像:
class SMSearchCriteria
{
public SMKeywords keywords { get; set; }
public SMDates dates { get; set; }
public SMClusters clusters { get; set; }
public SMTaxominies taxonomies { get; set; }
public SMSort sort { get; set; }
}
class SMTaxominies
{
public Dictionary<string, string> Topics;
public Keywords<string, string> Keywords;
}
我首先从一个强类型的dto类开始,首先是一个DataContract,你可以选择将它序列化为你想要的任何格式,即JSON,Xml,ProtoBuf等。
注意:与大多数其他格式相比,JSON实际上序列化/反序列化的速度相当慢(参见: 序列化基准测试 - JSON.NET是'NewtonSoft.Json' )如果您使用的是富客户端应用程序而不是Web应用程序可能想要选择不同的序列化格式。 无论您最终使用哪种格式,您仍然可以重复使用相同的DTO,例如上面的代码看起来像:
[DataContract]
public class MyDto
{
[DataMember]
public Keywords keywords { get; set; }
}
[DataContract]
public class Keywords
{
[DataMember]
public List<string> anyString { get; set; }
[DataMember]
public Dictionary<string,string> allString { get; set; }
[DataMember]
public List<string> exactString { get; set; }
[DataMember]
public List<string> notString { get; set; }
[DataMember]
public List<string> highlightString { get; set; }
}
var dto = new MyDto { Keywords = { allString = {{"a5349f533e3aa3ccbc27de2638da38d6", "olympics"}} };
var json = JsonConvert.SerializeObject(dto);
var fromJson = JsonConvert.DeserializeObject<MyDto>(json);
编辑:添加链接
看起来你可能遇到了JSON.NET的问题,在这种情况下你应该尝试其他JSON .NET序列化器/反序列化器。 Microsoft提供了.NET v3.5中包含的System.Runtime.Serialization.Json.DataContractJsonSerializer。 下面是一些帮助程序类,向您展示如何序列化和反序列化 JSON。
Jayrock是另一个用于.NET的JSON序列化程序,但它比其他程序慢,我发现它对Generics没有很好的支持。
你为什么不使用LINQ到Json?
例如,将“排序”节点映射到您的类
var jsonResult = JObject.Parse(jsonString);
var sortItem = from s in jsonResult["sort"]
select new MySortObject{
SortId = s.Value<int>("sortId")
};
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.