.NET JSONパーサーを使用していますが、構成ファイルをシリアル化して読みやすくしたいと考えています。代わりに:
{"blah":"v", "blah2":"v2"}
私は次のようなもっといいものが欲しいです:
{
"blah":"v",
"blah2":"v2"
}
私のコードは次のようなものです:
using System.Web.Script.Serialization;
var ser = new JavaScriptSerializer();
configSz = ser.Serialize(config);
using (var f = (TextWriter)File.CreateText(configFn))
{
f.WriteLine(configSz);
f.Close();
}
JavaScriptSerializerでこれを達成するのに苦労するでしょう。
JSON.Net を試してください。
JSON.Netの例から若干の変更を加えて
using System;
using Newtonsoft.Json;
namespace JsonPrettyPrint
{
internal class Program
{
private static void Main(string[] args)
{
Product product = new Product
{
Name = "Apple",
Expiry = new DateTime(2008, 12, 28),
Price = 3.99M,
Sizes = new[] { "Small", "Medium", "Large" }
};
string json = JsonConvert.SerializeObject(product, Formatting.Indented);
Console.WriteLine(json);
Product deserializedProduct = JsonConvert.DeserializeObject<Product>(json);
}
}
internal class Product
{
public String[] Sizes { get; set; }
public decimal Price { get; set; }
public DateTime Expiry { get; set; }
public string Name { get; set; }
}
}
結果
{
"Sizes": [
"Small",
"Medium",
"Large"
],
"Price": 3.99,
"Expiry": "\/Date(1230447600000-0700)\/",
"Name": "Apple"
}
ドキュメント:オブジェクトのシリアル化
Json.Netライブラリの短いサンプルコード
private static string FormatJson(string json)
{
dynamic parsedJson = JsonConvert.DeserializeObject(json);
return JsonConvert.SerializeObject(parsedJson, Formatting.Indented);
}
JSON文字列があり、それを「プリティ」したいが、既知のC#型との間でシリアル化したくない場合は、次の方法が役立ちます(JSON.NETを使用)。
using System;
using System.IO;
using Newtonsoft.Json;
class JsonUtil
{
public static string JsonPrettify(string json)
{
using (var stringReader = new StringReader(json))
using (var stringWriter = new StringWriter())
{
var jsonReader = new JsonTextReader(stringReader);
var jsonWriter = new JsonTextWriter(stringWriter) { Formatting = Formatting.Indented };
jsonWriter.WriteToken(jsonReader);
return stringWriter.ToString();
}
}
}
既存のJSONを指定:の最短バージョン(編集:JSON.netを使用)
JToken.Parse("mystring").ToString()
入力:
{"menu": { "id": "file", "value": "File", "popup": { "menuitem": [ {"value": "New", "onclick": "CreateNewDoc()"}, {"value": "Open", "onclick": "OpenDoc()"}, {"value": "Close", "onclick": "CloseDoc()"} ] } }}
出力:
{
"menu": {
"id": "file",
"value": "File",
"popup": {
"menuitem": [
{
"value": "New",
"onclick": "CreateNewDoc()"
},
{
"value": "Open",
"onclick": "OpenDoc()"
},
{
"value": "Close",
"onclick": "CloseDoc()"
}
]
}
}
}
オブジェクトのプリティプリント:
JToken.FromObject(myObject).ToString()
Newtonsoft.Json
を使用する1つのライナー:
string prettyJson = JToken.Parse(uglyJsonString).ToString(Formatting.Indented);
フォーマットされたJsonを取得するには、次の標準的な方法を使用できます
JsonReaderWriterFactory.CreateJsonWriter(ストリームストリーム、エンコーディングエンコーディング、bool ownsStream、boolインデント、文字列indentChars)
設定のみ"indent == true"
このようなものを試してください
public readonly DataContractJsonSerializerSettings Settings =
new DataContractJsonSerializerSettings
{ UseSimpleDictionaryFormat = true };
public void Keep<TValue>(TValue item, string path)
{
try
{
using (var stream = File.Open(path, FileMode.Create))
{
var currentCulture = Thread.CurrentThread.CurrentCulture;
Thread.CurrentThread.CurrentCulture = CultureInfo.InvariantCulture;
try
{
using (var writer = JsonReaderWriterFactory.CreateJsonWriter(
stream, Encoding.UTF8, true, true, " "))
{
var serializer = new DataContractJsonSerializer(type, Settings);
serializer.WriteObject(writer, item);
writer.Flush();
}
}
catch (Exception exception)
{
Debug.WriteLine(exception.ToString());
}
finally
{
Thread.CurrentThread.CurrentCulture = currentCulture;
}
}
}
catch (Exception exception)
{
Debug.WriteLine(exception.ToString());
}
}
行に注意してください
var currentCulture = Thread.CurrentThread.CurrentCulture;
Thread.CurrentThread.CurrentCulture = CultureInfo.InvariantCulture;
....
Thread.CurrentThread.CurrentCulture = currentCulture;
InvariantCultureを使用して、異なる地域設定を持つコンピューターでのシリアル化解除中の例外を回避する必要があります。たとえば、無効な形式のdoubleまたはDateTimeが原因である場合があります。
デシリアライズ用
public TValue Revive<TValue>(string path, params object[] constructorArgs)
{
try
{
using (var stream = File.OpenRead(path))
{
var currentCulture = Thread.CurrentThread.CurrentCulture;
Thread.CurrentThread.CurrentCulture = CultureInfo.InvariantCulture;
try
{
var serializer = new DataContractJsonSerializer(type, Settings);
var item = (TValue) serializer.ReadObject(stream);
if (Equals(item, null)) throw new Exception();
return item;
}
catch (Exception exception)
{
Debug.WriteLine(exception.ToString());
return (TValue) Activator.CreateInstance(type, constructorArgs);
}
finally
{
Thread.CurrentThread.CurrentCulture = currentCulture;
}
}
}
catch
{
return (TValue) Activator.CreateInstance(typeof (TValue), constructorArgs);
}
}
ありがとう!
最初に、Duncan Smartの投稿にコメントを追加したかったのですが、残念ながら、コメントを残すにはまだ十分な評判を得ていません。ここで試してみます。
副作用について警告したいだけです。
JsonTextReaderは、jsonを内部的に解析して型指定されたJTokenに変換し、その後シリアル化します。
たとえば、元のJSONが
{ "double":0.00002, "date":"\/Date(1198908717056)\/"}
プリティファイした後
{
"double":2E-05,
"date": "2007-12-29T06:11:57.056Z"
}
もちろん、両方のjson文字列は同等であり、構造的に等しいオブジェクトにデシリアライズされますが、元の文字列値を保持する必要がある場合は、これを考慮に入れる必要があります