C # Эквивалент PHP http_build_query

Мне нужно передать некоторые данные на страницу PHP на сервере с моего клиента C #, используя HttpWebRequest . Ожидаемые данные в соответствии с документацией представляют собой массив массивов, что-то вроде этого:

 $postData = array( 'label1' => 'myLabel', 'label2' => array( 'label2_1' => 3 'label2_2' => array( 'label2_2_1' => 3 ) ) ); 

Структура, приведенная выше, является лишь примером. Это может быть очень сложно, и сама структура не является постоянной.

В PHP есть функция с именем http_build_query которая сериализует эти вложенные PHP массивы простой строкой, которая может быть отправлена ​​как данные HTTP POST-запроса. Проблема в том, что мне нужно вызвать эту страницу PHP из моего приложения C #. Я хотел бы представить эти вложенные массивы либо в виде вложенного Dictionary<string, object> s, либо анонимных типов.

Как я могу это сделать? Какие правила выполняет http_build_query для создания выходной строки?

Существует очень похожий вопрос. Преобразование массива массивов PHP в C # , что, к сожалению, не решает мою проблему. В принятом ответе рекомендуется решение для фиксированной структуры, второе – вообще не работает.

Related of "C # Эквивалент PHP http_build_query"

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

Я взял подход «черного ящика» и создал следующий класс:

 /// <summary> /// Helps up build a query string by converting an object into a set of named-values and making a /// query string out of it. /// </summary> public class QueryStringBuilder { private readonly List<KeyValuePair<string, object>> _keyValuePairs = new List<KeyValuePair<string, object>>(); /// <summary> Builds the query string from the given instance. </summary> public static string BuildQueryString(object queryData, string argSeperator = "&") { var encoder = new QueryStringBuilder(); encoder.AddEntry(null, queryData, allowObjects: true); return encoder.GetUriString(argSeperator); } /// <summary> /// Convert the key-value pairs that we've collected into an actual query string. /// </summary> private string GetUriString(string argSeperator) { return String.Join(argSeperator, _keyValuePairs.Select(kvp => { var key = Uri.EscapeDataString(kvp.Key); var value = Uri.EscapeDataString(kvp.Value.ToString()); return $"{key}={value}"; })); } /// <summary> Adds a single entry to the collection. </summary> /// <param name="prefix"> The prefix to use when generating the key of the entry. Can be null. </param> /// <param name="instance"> The instance to add. /// /// - If the instance is a dictionary, the entries determine the key and values. /// - If the instance is a collection, the keys will be the index of the entries, and the value /// will be each item in the collection. /// - If allowObjects is true, then the object's properties' names will be the keys, and the /// values of the properties will be the values. /// - Otherwise the instance is added with the given prefix to the collection of items. </param> /// <param name="allowObjects"> true to add the properties of the given instance (if the object is /// not a collection or dictionary), false to add the object as a key-value pair. </param> private void AddEntry(string prefix, object instance, bool allowObjects) { var dictionary = instance as IDictionary; var collection = instance as ICollection; if (dictionary != null) { Add(prefix, GetDictionaryAdapter(dictionary)); } else if (collection != null) { Add(prefix, GetArrayAdapter(collection)); } else if (allowObjects) { Add(prefix, GetObjectAdapter(instance)); } else { _keyValuePairs.Add(new KeyValuePair<string, object>(prefix, instance)); } } /// <summary> Adds the given collection of entries. </summary> private void Add(string prefix, IEnumerable<Entry> datas) { foreach (var item in datas) { var newPrefix = String.IsNullOrEmpty(prefix) ? item.Key : $"{prefix}[{item.Key}]"; AddEntry(newPrefix, item.Value, allowObjects: false); } } private struct Entry { public string Key; public object Value; } /// <summary> /// Returns a collection of entries that represent the properties on the object. /// </summary> private IEnumerable<Entry> GetObjectAdapter(object data) { var properties = data.GetType().GetProperties(); foreach (var property in properties) { yield return new Entry() { Key = property.Name, Value = property.GetValue(data) }; } } /// <summary> /// Returns a collection of entries that represent items in the collection. /// </summary> private IEnumerable<Entry> GetArrayAdapter(ICollection collection) { int i = 0; foreach (var item in collection) { yield return new Entry() { Key = i.ToString(), Value = item, }; i++; } } /// <summary> /// Returns a collection of entries that represent items in the dictionary. /// </summary> private IEnumerable<Entry> GetDictionaryAdapter(IDictionary collection) { foreach (DictionaryEntry item in collection) { yield return new Entry() { Key = item.Key.ToString(), Value = item.Value, }; } } } 

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

Например, следующее:

 QueryStringBuilder.BuildQueryString(new { Age = 19, Name = "John&Doe", Values = new object[] { 1, 2, new Dictionary<string, string>() { { "key1", "value1" }, { "key2", "value2" }, } }, }); // 0=1&1=2&2%5B0%5D=one&2%5B1%5D=two&2%5B2%5D=three&3%5Bkey1%5D=value1&3%5Bkey2%5D=value2 QueryStringBuilder.BuildQueryString(new object[] { 1, 2, new object[] { "one", "two", "three" }, new Dictionary<string, string>() { { "key1", "value1" }, { "key2", "value2" }, } } ); 

Формирует:

 Age=19&Name=John%26Doe&Values%5B0%5D=1&Values%5B1%5D=2&Values%5B2%5D%5Bkey1%5D=value1&Values%5B2%5D%5Bkey2%5D=value2 

который:

 Age=19&Name=John%26Doe&Values[0]=1&Values[1]=2&Values[2][key1]=value1&Values[2][key2]=value2 Age=19 Name=John&Doe Values[0]=1 Values[1]=2 Values[2][key1]=value1 Values[2][key2]=value2 

Используя NameValueCollection, вы можете сделать это:

 private string ToQueryString(NameValueCollection queryData) { var array = (from key in queryData.AllKeys from value in queryData.GetValues(key) select string.Format(CultureInfo.InvariantCulture, "{0}={1}", HttpUtility.UrlEncode(key), HttpUtility.UrlEncode(value))) .ToArray(); return "?" + string.Join("&", array); }