C #을 사용하여 .NET에서 JSON 형식을 지정하려면 어떻게합니까?


256

.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();
}

답변:


257

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"
}

설명서 : 객체 직렬화



15
@Brad 그는 절대적으로 같은 코드를 보여 주었지만 모델을 사용했다
Mia

아이디어는 Formatting.Indented 그래서
FindOutIslamNow

이 방법을 사용하면 JSON 형식 오류가 발생하지 않습니다.
Anshuman Goel

173

Json.Net 라이브러리의 더 짧은 샘플 코드

private static string FormatJson(string json)
{
    dynamic parsedJson = JsonConvert.DeserializeObject(json);
    return JsonConvert.SerializeObject(parsedJson, Formatting.Indented);
}

1
실제로 한 단계 더 나아가 확장 방법을 만들 수 있습니다. 공개하고 서명을 FormatJson (이 문자열 json)으로 변경
bdwakefield

129

JSON 문자열이 있고 "prettify"하고 싶지만 알려진 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();
        }
    }
}

6
단지 JSON 문자열을 값 싸게 치장을 위해이 ... 다른 사람보다 훨씬 더 적합한 솔루션입니다
옌스 Marchewka

2
다음 사용 사례는 실패 JsonPrettify("null")하고JsonPrettify("\"string\"")
Ekevoo

1
감사합니다 @Ekevoo, 나는 그것을 이전 버전으로 롤백했습니다!
던컨 스마트

@DuncanSmart 나는 이것을 좋아한다! 그 버전은 더 적은 임시 객체를 만듭니다. 유스 케이스가 작동하더라도 비난했던 것보다 낫다고 생각합니다.
Ekevoo

97

기존 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()

4
이것은 json의 구조를 미리 몰라도 작동합니다. 그리고 가장 짧은 답변입니다.
foresightyj

1
이것은 작동하지만 json 객체가 배열이 아닌 경우에만 가능합니다. 그것이 배열이라는 것을 알고 있다면 대신 JArray.Parse를 사용할 수 있습니다.
Luke Z

3
아, 좋은 지적입니다. 감사합니다. JToken대신에 사용하도록 답변을 업데이트했습니다 JObject. 때문에 이것은, 객체 또는 배열 작동 JToken모두 상위 클래스입니다 JObject및이 JArray.
asherber 2018 년

고마워요,이 솔루션을 얻기 위해 약 2 시간을 낭비한 사람 ... @stackoverflow없이 내 인생을 상상할 수 없습니다 ...
Rudresha Parameshappa

나는 다른 답변보다 이것을 선호합니다. 짧은 코드와 효과적인. 감사합니다
Marc Roussel

47

사용하는 Oneliner Newtonsoft.Json.Linq:

string prettyJson = JToken.Parse(uglyJsonString).ToString(Formatting.Indented);

나는 이것이 Newtonsoft를 사용하여 JSON 형식을 지정하는 가장 간단한 API라는 것에 동의합니다.
Ethan Wu

2
Newtonsoft.Json에서 이것을 찾을 수 없습니다 ... 아마도 이전 버전이있을 수 있습니다.
cslotty 2012

2
NewtonSoft.Json.Linq 네임 스페이스에 있습니다. 나도 그것을 찾았 기 때문에 나는 이것을 알고있다.
켄 파치 선장

12

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;

일부 종류의 xml-serializer의 경우 InvariantCulture 를 사용 하여 지역 설정이 다른 컴퓨터에서 deserialization하는 동안 예외를 피해야합니다. 예를 들어, 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);
        }
    }

감사!


안녕하세요, @Makeman, 다른 문화로 인한 직렬화 오류를 재현 한 적이 있습니까? XmlJsonWriter / Reader 변환은 모두 문화적으로 변하지 않는 것 같습니다.
Olexander Ivanitskyi

안녕하세요, XmlJsonWriter / Reader에 대해 잘 모르지만 DataContractJsonSerializer는 Thread.CurrentThread.CurrentCulture를 사용합니다. 머신 A에서 데이터가 직렬화되었지만 다른 지역 설정으로 B에서 직렬화 해제 된 경우 오류가 발생할 수 있습니다.
Makeman

DataContractJsonSerializer어셈블리 System.Runtime.Serialization v.4.0.0.0에서 디 컴파일 했는데 , 명시적인 사용법이 없습니다 CurrentCulture. 문화권의 유일한 사용법은 CultureInfo.InvariantCulture기본 클래스 XmlObjectSerializer의 내부 메서드 TryAddLineInfo입니다.
Olexander Ivanitskyi

아마도 내 실수일지도 모른다. 나중에 확인하겠습니다. 가능한 한 다른 직렬 변환기를 구현 하여이 문화 문제를 추정합니다.
Makeman

1
원래 답변을 편집했습니다. DataContract 직렬 변환기는 배양에 독립적 인 것으로 보이지만 다른 종류의 직렬 변환기에 의한 직렬화 중에 문화 별 오류를 피하려면주의를 기울여야합니다. :)
Makeman

6

이 모든 것은 한 줄로 이루어질 수 있습니다 :

string jsonString = JsonConvert.SerializeObject(yourObject, Formatting.Indented);

1
'Newtonsoft.Json 사용'
Ebube

내 친구에게 가장 답하십시오.
RogerEdward

5

다음은 Microsoft의 System.Text.Json 라이브러리를 사용하는 솔루션입니다 .

static string FormatJsonText(string jsonString)
{
    using var doc = JsonDocument.Parse(
        jsonString,
        new JsonDocumentOptions
        {
            AllowTrailingCommas = true
        }
    );
    MemoryStream memoryStream = new MemoryStream();
    using (
        var utf8JsonWriter = new Utf8JsonWriter(
            memoryStream,
            new JsonWriterOptions
            {
                Indented = true
            }
        )
    )
    {
        doc.WriteTo(utf8JsonWriter);
    }
    return new System.Text.UTF8Encoding()
        .GetString(memoryStream.ToArray());
}

추가 패키지를 구매할 수없는 사람들에게 좋은 솔루션입니다. 잘 작동합니다.
Mark T

2

먼저 Duncan Smart 게시물에 의견을 추가하고 싶었지만 안타깝게도 아직 의견을 남길만한 명성이 없습니다. 그래서 나는 여기서 시도 할 것입니다.

부작용에 대해서만 경고하고 싶습니다.

JsonTextReader는 내부적으로 json을 유형이 지정된 JToken으로 구문 분석 한 후 다시 직렬화합니다.

예를 들어 원래 JSON이

 { "double":0.00002, "date":"\/Date(1198908717056)\/"}

prettify 후 얻을

{ 
    "double":2E-05,
    "date": "2007-12-29T06:11:57.056Z"
}

물론 두 json 문자열은 동일하며 구조적으로 동일한 객체로 직렬화 해제되지만 원래 문자열 값을 유지 해야하는 경우 이것을 조정해야합니다.


여기 세부 ...에 대해 많은 논의가 github.com/JamesNK/Newtonsoft.Json/issues/862의 이 세부가 진화하는 방법을 흥미. 내 기본 json 파서에 대해 새로운 것을 배웠습니다. 귀하의 의견에 감사드립니다.
Sql Surfer

2

System.Text.Jsonset 사용 JsonSerializerOptions.WriteIndented = true:

JsonSerializerOptions options = new JsonSerializerOptions { WriteIndented = true };
string json = JsonSerializer.Serialize<Type>(object, options);

2

netcoreapp3.1

var js = JsonSerializer.Serialize(obj, new JsonSerializerOptions {
             WriteIndented = true
         });

0

이것은 나를 위해 일했습니다. 누군가 VB.NET 버전을 찾고있는 경우.

@imports System
@imports System.IO
@imports Newtonsoft.Json

Public Shared Function JsonPrettify(ByVal json As String) As String
  Using stringReader = New StringReader(json)

    Using stringWriter = New StringWriter()
      Dim jsonReader = New JsonTextReader(stringReader)
      Dim jsonWriter = New JsonTextWriter(stringWriter) With {
          .Formatting = Formatting.Indented
      }
      jsonWriter.WriteToken(jsonReader)
      Return stringWriter.ToString()
    End Using
  End Using
End Function

0

아래 코드는 저에게 효과적입니다.

JsonConvert.SerializeObject(JToken.Parse(yourobj.ToString()))
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.