Подтвердить что ты не робот

ASP.NET MVC - передать объект массива как значение маршрута в Html.ActionLink(...)

У меня есть метод, который возвращает массив (string []), и я пытаюсь передать этот массив строк в Action Link, чтобы он создал строку запроса, похожую на:

/Controller/Action?str=val1&str=val2&str=val3...etc

Но когда я передаю новый {str = GetStringArray()}, я получаю следующий URL:

/Controller/Action?str=System.String%5B%5D

Итак, в основном это берет мою строку [] и запускает .ToString() на ней, чтобы получить значение.

Любые идеи? Спасибо!

4b9b3361

Ответ 1

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

<%  var rv = new RouteValueDictionary();
    var strings = GetStringArray();
    for (int i = 0; i < strings.Length; ++i)
    {
        rv["str[" + i + "]"] = strings[i];
    }
 %>

<%= Html.ActionLink( "Link", "Action", "Controller", rv, null ) %>

предоставит вам ссылку, например

<a href='/Controller/Action?str=val0&str=val1&...'>Link</a>

EDIT: MVC2 изменил интерфейс ValueProvider, чтобы мой оригинальный ответ устарел. Вы должны использовать модель с массивом строк как свойство.

public class Model
{
    public string Str[] { get; set; }
}

Затем модельное связующее заполнит вашу модель значениями, которые вы передаете в URL.

public ActionResult Action( Model model )
{
    var str0 = model.Str[0];
}

Ответ 2

Это действительно раздражало меня, так что вдохновение от Scott Hanselman Я написал следующий (плавный) метод расширения:

public static RedirectToRouteResult WithRouteValue(
    this RedirectToRouteResult result, 
    string key, 
    object value)
{
    if (value == null)
        throw new ArgumentException("value cannot be null");

    result.RouteValues.Add(key, value);

    return result;
}

public static RedirectToRouteResult WithRouteValue<T>(
    this RedirectToRouteResult result, 
    string key, 
    IEnumerable<T> values)
{
    if (result.RouteValues.Keys.Any(k => k.StartsWith(key + "[")))
        throw new ArgumentException("Key already exists in collection");

    if (values == null)
        throw new ArgumentNullException("values cannot be null");

    var valuesList = values.ToList();

    for (int i = 0; i < valuesList.Count; i++)
    {
        result.RouteValues.Add(String.Format("{0}[{1}]", key, i), valuesList[i]);
    }

    return result;
}

Вызываем так:

return this.RedirectToAction("Index", "Home")
           .WithRouteValue("id", 1)
           .WithRouteValue("list", new[] { 1, 2, 3 });

Ответ 3

Другое решение, которое мне пришло в голову:

string url = "/Controller/Action?iVal=5&str=" + string.Join("&str=", strArray); 

Это грязно, и вы должны проверить его перед его использованием, но он должен работать, тем не менее. Надеюсь, это поможет.

Ответ 4

Существует библиотека под названием Unbinder, которую вы можете использовать для вставки сложных объектов в маршруты /URL.

Он работает следующим образом:

using Unbound;

Unbinder u = new Unbinder();
string url = Url.RouteUrl("routeName", new RouteValueDictionary(u.Unbind(YourComplexObject)));

Ответ 5

Это массив решений HelperExtension и проблемы, связанные с IEnumerable:

public static class AjaxHelperExtensions
{
    public static MvcHtmlString ActionLinkWithCollectionModel(this AjaxHelper ajaxHelper, string linkText, string actionName, object model, AjaxOptions ajaxOptions, IDictionary<string, object> htmlAttributes)
    {
        var rv = new RouteValueDictionary();

        foreach (var property in model.GetType().GetProperties())
        {
            if (typeof(ICollection).IsAssignableFrom(property.PropertyType))
            {
                var s = ((IEnumerable<object>)property.GetValue(model));
                if (s != null && s.Any())
                {
                    var values = s.Select(p => p.ToString()).Where(p => !string.IsNullOrEmpty(p)).ToList();
                    for (var i = 0; i < values.Count(); i++)
                        rv.Add(string.Concat(property.Name, "[", i, "]"), values[i]);
                }
            }
            else
            {
                var value = property.GetGetMethod().Invoke(model, null) == null ? "" : property.GetGetMethod().Invoke(model, null).ToString();
                if (!string.IsNullOrEmpty(value))
                    rv.Add(property.Name, value);
            }
        }
        return System.Web.Mvc.Ajax.AjaxExtensions.ActionLink(ajaxHelper, linkText, actionName, rv, ajaxOptions, htmlAttributes);
    }
}

Ответ 6

Я бы использовал POST для массива. Помимо уродства и злоупотребления GET, вы рискуете исчерпать пространство URL (верьте или нет).

Предполагая ограничение в 2000 байт. Накладные расходы строки запроса (& str =) уменьшают до ~ 300 байтов фактических данных (при условии, что остальная часть URL-адреса равна 0 байтам).