Хранение анонимного объекта в ViewBag

Это, наверное, глупый вопрос, но я пытаюсь заполнить анонимный объект ViewBag вот так:

ViewBag.Stuff = new { Name = "Test", Email = "user@domain.com" };

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

@ ViewBag.Stuff.Name

Я понимаю, что ViewBag является динамическим, и что Stuff - это анонимный объект... но когда я смотрю с помощью отладчика из строки View выше, я вижу все свойства с правильными значениями. Почему связующему модели так тяжело с этим?

Есть ли хороший способ сделать это без создания модельного класса? Я хочу продолжать использовать new {}

3 ответа

Решение

По сути, проблема заключается в том, что анонимные типы генерируются как внутренние ( см. Ответ), что делает невозможным типизированные ссылки на свойство объекта из представления. Эта статья предоставляет более подробное объяснение:

http://www.heartysoft.com/anonymous-types-c-sharp-4-dynamic

Этого можно добиться с помощью класса-оболочки Dynamic Anonymous (ответ @Dakill), но он становится ужасно быстрым, и у программиста возникает вопрос, почему он / она так поступил.

Вопреки распространенному мнению, это может быть сделано, но включает в себя несколько уродливый хак, который приведет к проблемам с обслуживанием в будущем. Он включает в себя написание класса, который "обернет" ваш анонимный объект в динамический объект. Я сделал это как упражнение некоторое время назад, ниже приведен код класса-обертки, вы бы использовали его как ViewBag.Stuff = new DynamicAnonymous(new { Name = "Test", Email = "user@domain.com" });..

public class DynamicAnonymous : DynamicObject
{
            object obj;

            public DynamicAnonymous(object o)
            {
                    this.obj = o;
            }

            public override IEnumerable<string> GetDynamicMemberNames()
            {
                    return obj.GetType().GetProperties().Select(n => n.Name);
            }

            public override bool TryGetMember(GetMemberBinder binder, out object result)
            {
                    var prop = obj.GetType().GetProperty(binder.Name);
                    if (prop == null)
                    {
                            result = null;
                            return false;
                    }
                    else
                    {
                            result = prop.GetValue(obj, null);
                            return true;
                    }
            }

            public override int GetHashCode()
            {
                    return obj.GetHashCode();
            }

            public override string ToString()
            {
                    return obj.ToString();
            }

            public override bool Equals(object obj2)
            {
                    return obj.Equals(obj2);
            }                
 }

Мы можем сделать это с помощью Json

Контроллер: объект -> строка json, Вид: строка Json -> объект

Scennerio - это просто класс контроллера, сериализующий объект C# в строку json, а затем представление получает эту строку и десериализует ее в объект следующим образом:

в контроллере:

using Newtonsoft.Json;
ViewBag.Stuff = JsonConvert.SerializeObject(new { Name = "Test", Email = "user@domain.com" });

ввиду:

@using Newtonsoft.Json
<p>@JsonConvert.DeserializeObject(ViewBag.Stuff).Name</p>

Примечание: это было проверено в Asp.Net Core 2.2, проверьте эту ссылку, чтобы установить Newtonsoft.Json

Вы можете сделать это, используя механизм NothingsImpossible descibed, но без реализации собственной оболочки, используя ExpandoObject. Вот пример:

var items = _repository.GetItems()
    .Select(og => {
        dynamic eo = new System.Dynamic.ExpandoObject();
        eo.Id = item.Id;
        eo.FriendlyName = og.FriendlyName;
        eo.Selected = itemIds.Contains(item.Id);
        return eo;
    })
    .ToArray();
ViewBag.Items = items;
Другие вопросы по тегам