3

次のように (Newtonsoft を使用して) JSON データを逆シリアル化し、C# で厳密に型指定されたオブジェクト/リストに変換したいのですが、インデックス付きの参照が参照されたオブジェクトに変換されるようにクラスを定義する方法がわかりません。

{
  "Countries": [
    {
      "Name": "USA",

    },
    {
      "Name": "UK",
    },
    {
      "Name": "JAPAN",
    },
  ],
  "Authors": [
    {
      "DisplayName": "John Doe",
      "RealName": "Not John Doe"
    },
    {
      "DisplayName": "Jane Doe",
      "RealName": "Not Jane Doe"
    },
  ],
  "Books": [
    {
      "Author": 0,
      "Title": "A good read",
      "PublishedCountries": "0,1",

    },
    {
      "Author": 0,
      "Title": "Another good read",
      "PublishedCountries": "0,1",
    },
    {
      "Author": 1,
      "Title": "This one is even better",
      "PublishedCountries": "0,1,2",
    },
  ],
}

理想的には、次のようなクラスを使用したいと思います。

public class Country
{
  public string Name { get; set;}
}

public class AuthorDetails
{
  public string DisplayName { get; set; }
  public string RealName { get; set; }
}

public class Book
{
  public AuthorDetails Author { get; set; }
  public string Title { get; set; }
  public IEnumerable<Country> PublishedCountries { get; set; }
}

public class ListOfBooks
{
  public IEnumerable<Book> Books { get; set; }
}

そして、次のように逆シリアル化します。

var listOfBooks = JsonConvert.DeserializeObject<ListOfBooks>(jsonAsString);

本の JObject の Author プロパティが整数ではなくインデックスであることを Json.Net に伝える方法に行き詰まっています。PublishedCountries についても同様です (インデックスのカンマ区切りのリストです)。

4

2 に答える 2

1

逆シリアル化プロセスを少し支援する以外に方法がわかりません。

var dynObj = (JObject)JsonConvert.DeserializeObject(json);

var authors = dynObj["Authors"]
                .Select(j => new AuthorDetails { 
                            RealName = (string)j["RealName"], 
                            DisplayName = (string)j["DisplayName"] 
                })
                .ToList();

var countries = dynObj["Countries"]
                .Select(j => new Country { Name = (string)j["Name"]})
                .ToList();

var books = dynObj["Books"].Select(x => new Book
                {
                    Author = authors[(int)x["Author"]],
                    Title = (string)x["Title"],
                    PublishedCountries = x["PublishedCountries"].ToString().Split(',')
                                            .Select(i =>countries[int.Parse(i)])
                                            .ToList()
                })
                .ToList();

public class Country
{
    public string Name { get; set; }
}

public class AuthorDetails
{
    public string DisplayName { get; set; }
    public string RealName { get; set; }
}

public class Book
{
    public AuthorDetails Author { get; set; }
    public string Title { get; set; }
    public List<Country> PublishedCountries { get; set; }
}
于 2012-10-09T09:39:30.863 に答える
0

あなたがしたいことは「オブジェクト参照を保持する」ことだと思います。

オブジェクトのコンテンツを最初から複製するのではなく、C# オブジェクトを属性でマークアップして、いつ参照を利用するかを記述することができます。

[JsonObject(IsReference = true)](型宣言で使用)

ただし、生成された Javascript は、デモしたものとはまったく異なります。ここを見て、その方法を確認してください。

http://james.newtonking.com/projects/json/help/?topic=html/PreserveObjectReferences.htm

代わりに、JSON は次のようになります。

{
  "Countries": [
    {
      "Name": "USA",

    },
    {
      "Name": "UK",
    },
    {
      "Name": "JAPAN",
    },
  ],
  "Authors": [
    {
      "DisplayName": "John Doe",
      "RealName": "Not John Doe"
    },
    {
      "DisplayName": "Jane Doe",
      "RealName": "Not Jane Doe"
    },
  ],
  "Books": [
    {
      "$id": 1,
      "Author": 0,
      "Title": "A good read",
      "PublishedCountries": "0,1",

    },
    {
      "$ref": 1
    },
    {
      "Author": 1,
      "Title": "This one is even better",
      "PublishedCountries": "0,1,2",
    },
  ],
}
于 2012-10-09T03:39:17.350 に答える