C# – анализ данных в формате JSON во вложенные хэш-таблицы

StackOverflow https://stackoverflow.com/questions/802766

  •  03-07-2019
  •  | 
  •  

Вопрос

Я пытаюсь работать с некоторыми данными в формате JSON на C#, но у меня возникли некоторые проблемы с определением правильного подхода к проблеме.Моя проблема в том, что данные в формате JSON будут в неизвестном формате (я знаю, это звучит странно… пожалуйста, читайте дальше).По сути, данные в формате json представляют собой некоторый набор пар имя/значение, где значения могут быть или не быть массивами вложенных пар имя/значение.Чтобы было веселее, вложение массивов пар имя/значение можно продолжать до бесконечности.

Например:Возможно, у меня есть какие-то данные, похожие на…

{
    "1": {
        "1.1": {
            "1.1.1": "value1",
            "1.1.2": "value2",
            "1.1.3": "value3"
        },
        "1.2": "value4",
        "1.3": {
            "1.3.1": {
                "1.3.1.1": "value5",
                "1.3.1.2": "value6"
            },
            "1.3.1.2": "value7",
            "1.3.1.3": "value8"
        }
    }
}

К сожалению, я не знаю, насколько будет вложенность, и технически я не знаю, какие пары имя/значение будут присутствовать в том или ином сообщении.

Есть ли в C# какой-либо поддерживаемый механизм, который позволил бы мне легко проанализировать это во вложенный набор hastables?

Я хотел бы сделать что-то вроде (обратите внимание, что этот код не на 100% синтаксически корректен, и его лучше было бы сделать с помощью рекурсии… но он передает идею).

Hashtable ht = [deserialize data method](jsonformattedstring);
foreach (Hashtable nested in ht)
{
    If (nested.count > 1)
        {
        Foreach (hashtable next in nested)
        …
        }
}
Это было полезно?

Решение

Мне не понравился анализ .Net Json... иногда он делает некоторые странные вещи.я перешел на Json.NET, библиотека с открытым исходным кодом.У него есть хороший объект JObject, который сделает то, что вам нужно.

Другие советы

В .NET у вас есть JsonArray, который позволяет загружать и анализировать данные JSON.Он создает массив JsonValue и полностью вложен на основе анализируемых данных JSON.

Если вам конкретно нужна Hashtable, вы можете перевести данные из JsonArray, хотя Hastable практически устарел в пользу Dictionary.

У Джоша Холмса есть довольно хороший пост о начале работы с JSON в .NET:http://www.joshholmes.com/blog/2009/01/20/PlayingWithJSON.aspx

Возможно, вы захотите посмотреть http://techblog.procurios.nl/k/n618/news/view/14605/14863/How-do-I-write-my-own-parser-for-JSON.html Это простая библиотека, которая анализирует строку JSON в Hashtables и ArrayLists.Он также может снова превратить эти структуры в JSON.

Вот метод, который я написал на C# для анализа JSON и возврата словаря.Конечно, это подходит не для всех случаев использования, но что-то вроде этого даст вам хороший однопроходный анализ JSON:

/*
     * This method takes in JSON in the form returned by javascript's
     * JSON.stringify(Object) and returns a string->string dictionary.
     * This method may be of use when the format of the json is unknown.
     * You can modify the delimiters, etc pretty easily in the source
     * (sorry I didn't abstract it--I have a very specific use).
     */ 
    public static Dictionary<string, string> jsonParse(string rawjson)
    {
        Dictionary<string, string> outdict = new Dictionary<string, string>();
        StringBuilder keybufferbuilder = new StringBuilder();
        StringBuilder valuebufferbuilder = new StringBuilder();
        StringReader bufferreader = new StringReader(rawjson);

        int s = 0;
        bool reading = false;
        bool inside_string = false;
        bool reading_value = false;
        //break at end (returns -1)
        while (s >= 0)
        {
            s = bufferreader.Read();
            //opening of json
            if (!reading)
            {
                if ((char)s == '{' && !inside_string && !reading) reading = true;
                continue;
            }
            else
            {
                //if we find a quote and we are not yet inside a string, advance and get inside
                if (!inside_string)
                {
                    //read past the quote
                    if ((char)s == '\"') inside_string = true;
                    continue;
                }
                if (inside_string)
                {
                    //if we reached the end of the string
                    if ((char)s == '\"')
                    {
                        inside_string = false;
                        s = bufferreader.Read(); //advance pointer
                        if ((char)s == ':')
                        {
                            reading_value = true;
                            continue;
                        }
                        if (reading_value && (char)s == ',')
                        {
                            //we know we just ended the line, so put itin our dictionary
                            if (!outdict.ContainsKey(keybufferbuilder.ToString())) outdict.Add(keybufferbuilder.ToString(), valuebufferbuilder.ToString());
                            //and clear the buffers
                            keybufferbuilder.Clear();
                            valuebufferbuilder.Clear();
                            reading_value = false;
                        }
                        if (reading_value && (char)s == '}')
                        {
                            //we know we just ended the line, so put itin our dictionary
                            if (!outdict.ContainsKey(keybufferbuilder.ToString())) outdict.Add(keybufferbuilder.ToString(), valuebufferbuilder.ToString());
                            //and clear the buffers
                            keybufferbuilder.Clear();
                            valuebufferbuilder.Clear();
                            reading_value = false;
                            reading = false;
                            break;
                        }
                    }
                    else
                    {
                        if (reading_value)
                        {
                            valuebufferbuilder.Append((char)s);
                            continue;
                        }
                        else
                        {
                            keybufferbuilder.Append((char)s);
                            continue;
                        }
                    }
                }
                else
                {
                    switch ((char)s)
                    {
                        case ':':
                            reading_value = true;
                            break;
                        default:
                            if (reading_value)
                            {
                                valuebufferbuilder.Append((char)s);
                            }
                            else
                            {
                                keybufferbuilder.Append((char)s);
                            }
                            break;
                    }
                }
            }
        }
        return outdict;
    }
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top