Хранение анонимного объекта в 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;