В С# преобразовать анонимный тип в массив key/value?

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

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

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

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

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

Ответ 1

Для выполнения этого требуется всего лишь небольшое отражение.

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);

Ответ 2

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

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

Ответ 3

Вот как они это делают в 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

Ответ 4

Решение @kbrimington делает хороший метод расширения - мой мой случай возвращает 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 и зациклился на результатах, но это намного опрятно.

Ответ 5

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

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

Ответ 6

Основываясь на @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 + "=" + HttpUtility.HtmlEncode(pair.Value).Replace(" ", "+"));
        } 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"

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