Я люблю простые костыли.

Когда требуется сериализовать некоторые поля в какой-то текстовый формат,
бывает удобно использовать промежуточное представление данных вида:

// Name/Value item
public class NVItem {
  public string? Name;
  public object? Value;
  public IEnumerable<NVItem>? SubItems;

  public NVItem(string? name, object? value) {
    Name = name;
    if (value==null) return;
    SubItems = value as IEnumerable<NVItem>;
    if (SubItems != null) return;
    
    // Упс!
    Value = value as string;
    if (Value != null) return;
    
    var num = value as IEnumerable;
    if (num==null) {
      Value = value;
    } else {
      SubItems = num.Cast<object>().Select(t => new NVItem(null, t));
    }
  }
}

Которое затем конвертируется в XML, JSON или что-нибудь ещё.

Например, так:

IEnumerable<NVItem> GetTypeInfo(Type type) {
  return new NVItem[] {
    new NVItem("Name", type.Name),
    new NVItem("FullName", type.FullName),
    new NVItem("CustomAttributes", type.CustomAttributes),
  };
}

Это можно реализовать и по-изящнее, но костыль улучшать - только портить.

Если требуется преобразовать данное представление в текст, воспользуемся следующими методами:

public static class NVItemExtensions {
  
  public static IEnumerable<string> ToStrings(this NVItem item) {
    yield return item.Name+'='+item.Value?.ToString();
    if (item.SubItems!=null) {
      foreach (string s in item.SubItems.SelectMany(t => t.ToStrings())) {
        yield return "    "+s;
      }
    }
  }
  
  public static string AsString(this IEnumerable<NVItem> items) { 
    return string.Join('\n', items.SelectMany(t => t.ToStrings()));
  }
}

string info = GetTypeInfo(typeof(string)).AsString();

Получаем:

Name=String
FullName=System.String
CustomAttributes=
    =[System.SerializableAttribute()]
    =[System.Runtime.CompilerServices.NullableContextAttribute((Byte)1)]
    =[System.Runtime.CompilerServices.NullableAttribute((Byte)0)]
    =[System.Reflection.DefaultMemberAttribute("Chars")]
    =[System.Runtime.Versioning.NonVersionableAttribute()]
    ...

Теперь заметим, что если:
- наши Name не начинаются с пробела и не содержат символов '=' и '\n',
- наши Value не содержат символов '\n',
- и нам не надо различать пустые строки и null,
то получаемый текст можно преобразовать обратно, за тем исключением, что Value теперь строки.

У меня не было такой надобности, но если сам пишешь - сам читаешь, то это вполне рабочий вариант сериализации. А вот сохранять настройки в подобный файл с намерением редактировать вручную, это не лучший выбор.

Форматы конфигов, как и все другие решения для всего, придумывают злые гении с подобающими им амбициями:
Habr: Tree — убийца JSON, XML, YAML и иже с ними
Habr: Почему JSON и YAML мешают вам писать нормальные конфиги (и чем их заменить)
Habr: JSON? JSONB? BSON? CBOR? MsgPack? А, VaryPackǃ


Комментарии (12)


  1. yarosroman
    20.01.2026 05:02

    Вы перепутали habr и govnokod.ru.


    1. AmirYantimirov Автор
      20.01.2026 05:02

      Обоснуйте. Вы имеете полное право считать, что этой статье здесь не место, но код делает в точности то, что я от него хочу.


      1. WhiteBehemoth
        20.01.2026 05:02

        if (true.ToString().ToLower() == "true") {...} тоже делает, то, что автор хочет. Вопрос - "зачем"?

        Я честно попытался прикинуть, в каком случае может понадобится ваш костыль - но ничего так и не придумалось. Можете дать пример, когда ваш костыль даёт хоть какую-то пользу перед другими сериализациями?


  1. yarosroman
    20.01.2026 05:02

    Ну для key\value существует Dictionary, который сереализуется в JSON и XML без всяких костылей, а если нужны Subitems, то легко положить в словарь и IEnumerable. Костыли они и на то костыли, что ими не хвастаются.


    1. AmirYantimirov Автор
      20.01.2026 05:02

      Как минимум, Dictionary не сохраняет порядок добавления.


      1. yarosroman
        20.01.2026 05:02

        OrderedDictionary


        1. AmirYantimirov Автор
          20.01.2026 05:02

          Да.

          А как насчет вложенности?


          1. yarosroman
            20.01.2026 05:02

            Можно спокойно добавить любую коллекцию в словарь.


            1. AmirYantimirov Автор
              20.01.2026 05:02

              Вложенность больше двух уровней.


              1. dopusteam
                20.01.2026 05:02

                Напишите все вводные сразу, а не по одному сообщению

                Кто мешает делать вложенность больше двух уровней, кстати?


          1. vabka
            20.01.2026 05:02

            OrderedDictionary<Key, object?>

            Но как уже выше сказали - нужны все вводные.


            1. AmirYantimirov Автор
              20.01.2026 05:02

              Вообще-то, я всегда сериализовал в XML вручную, и когда сейчас глянул в документацию, понял, что не представляю возможностей штатной сериализации стандартных контейнеров.

              Могли бы вы представить этот пример через Dictionary, и каков будет результат сериализации в XML?

              IEnumerable GetTypeInfo(Type type) {
              return new NVItem[] {
              new NVItem("Name", type.Name),
              new NVItem("FullName", type.FullName),
              new NVItem("CustomAttributes", type.CustomAttributes),
              };
              }