В c # конвертировать анонимный тип в массив ключ / значение? - PullRequest
73 голосов
/ 14 августа 2010

У меня есть следующий анонимный тип:

new {data1 = "test1", data2 = "sam", data3 = "bob"}

Мне нужен метод, который будет принимать это и выводить пары значений ключей в массиве или словаре.

Моя цель - использовать это как данные поста в HttpRequest, так что я в конечном итоге объединю в следующую строку:

"data1=test1&data2=sam&data3=bob"

Ответы [ 8 ]

111 голосов
/ 14 августа 2010

Для этого требуется лишь немного размышлений.

var a = new { data1 = "test1", data2 = "sam", data3 = "bob" };
var type = a.GetType();
var props = type.GetProperties();
var pairs = props.Select(x => x.Name + "=" + x.GetValue(a, null)).ToArray();
var result = string.Join("&", pairs);
61 голосов
/ 14 августа 2010

Если вы используете .NET 3.5 SP1 или .NET 4, вы можете (ab) использовать для этого RouteValueDictionary. Он реализует IDictionary<string, object> и имеет конструктор, который принимает object и преобразует свойства в пары ключ-значение.

Тогда было бы тривиально перебирать ключи и значения для построения строки запроса.

25 голосов
/ 14 августа 2010

Вот как они это делают в RouteValueDictionary:

  private void AddValues(object values)
    {
        if (values != null)
        {
            foreach (PropertyDescriptor descriptor in TypeDescriptor.GetProperties(values))
            {
                object obj2 = descriptor.GetValue(values);
                this.Add(descriptor.Name, obj2);
            }
        }
    }

Полный источник здесь: http://pastebin.com/c1gQpBMG

1 голос
/ 21 июля 2018

Существует встроенный метод преобразования анонимных объектов в словари:

HtmlHelper.AnonymousObjectToHtmlAttributes(yourObj)

Он также возвращает RouteValueDictionary. Обратите внимание, что это статический

1 голос
/ 19 апреля 2017

Основываясь на предложении @ GWB об использовании RouteValueDictionary, я написал эту рекурсивную функцию для поддержки вложенных анонимных типов, добавляя префикс этих вложенных параметров с помощью ключей их родителей.

public static string EncodeHtmlRequestBody(object data, string parent = null) {
    var keyValuePairs = new List<string>();
    var dict = new RouteValueDictionary(data);

    foreach (var pair in dict) {
        string key = parent == null ? pair.Key : parent + "." + pair.Key;
        var type = pair.Value.GetType();
        if (type.IsPrimitive || type == typeof(decimal) || type == typeof(string)) {
            keyValuePairs.Add(key + "=" + Uri.EscapeDataString((string)pair.Value).Replace("%20", "+"));
        } else {
            keyValuePairs.Add(EncodeHtmlRequestBody(pair.Value, key));
        }
    }

    return String.Join("&", keyValuePairs);
}

Пример использования:

var data = new {
    apiOperation = "AUTHORIZE",
    order = new {
        id = "order123",
        amount = "101.00",
        currency = "AUD"
    },
    transaction = new {
        id = "transaction123"
    },
    sourceOfFunds = new {
        type = "CARD",
        provided = new {
            card = new {
                expiry = new {
                    month = "1",
                    year = "20"
                },
                nameOnCard = "John Smith",
                number = "4444333322221111",
                securityCode = "123"
            }
        }
    }
};

string encodedData = EncodeHtmlRequestBody(data);

encodedData становится:

"apiOperation=AUTHORIZE&order.id=order123&order.amount=101.00&order.currency=AUD&transaction.id=transaction123&sourceOfFunds.type=CARD&sourceOfFunds.provided.card.expiry.month=1&sourceOfFunds.provided.card.expiry.year=20&sourceOfFunds.provided.card.nameOnCard=John+Smith&sourceOfFunds.provided.card.number=4444333322221111&sourceOfFunds.provided.card.securityCode=123"

Надеюсь, это поможет кому-то еще в подобной ситуации.

Редактировать: Как указывал DrewG, он не поддерживает массивы. Правильно реализовать поддержку произвольно вложенных массивов с анонимными типами было бы нетривиально, и поскольку ни один из API, которые я использовал, также не принимал массивы (я не уверен, что существует даже стандартизированный способ сериализации их с кодированием формы), Я оставлю это вам, ребята, если вам нужно их поддержать.

1 голос
/ 19 февраля 2015

Я сделал что-то вроде этого:

public class ObjectDictionary : Dictionary<string, object>
{
    /// <summary>
    /// Construct.
    /// </summary>
    /// <param name="a_source">Source object.</param>
    public ObjectDictionary(object a_source)
        : base(ParseObject(a_source))
    {

    }

    /// <summary>
    /// Create a dictionary from the given object (<paramref name="a_source"/>).
    /// </summary>
    /// <param name="a_source">Source object.</param>
    /// <returns>Created dictionary.</returns>
    /// <exception cref="ArgumentNullException">Thrown if <paramref name="a_source"/> is null.</exception>
    private static IDictionary<String, Object> ParseObject(object a_source)
    {
        #region Argument Validation

        if (a_source == null)
            throw new ArgumentNullException("a_source");

        #endregion

        var type = a_source.GetType();
        var props = type.GetProperties();

        return props.ToDictionary(x => x.Name, x => x.GetValue(a_source, null));
    }
}
1 голос
/ 30 января 2011

@ решение Кбримингтона делает хороший метод расширения - в моем случае я возвращаю HtmlString

    public static System.Web.HtmlString ToHTMLAttributeString(this Object attributes)
    {
        var props = attributes.GetType().GetProperties();
        var pairs = props.Select(x => string.Format(@"{0}=""{1}""",x.Name,x.GetValue(attributes, null))).ToArray();
        return new HtmlString(string.Join(" ", pairs));
    }

. Я использую его для перетаскивания произвольных атрибутов в представление Razor MVC.Я начал с кода, используя RouteValueDictionary и зацикливание на результатах, но это гораздо лучше.

0 голосов
/ 08 мая 2018
using Newtonsoft.Json;
var data = new {data1 = "test1", data2 = "sam", data3 = "bob"};
var encodedData = new FormUrlEncodedContent(JsonConvert.DeserializeObject<Dictionary<string, string>>(JsonConvert.SerializeObject(data))
...