使用ViewBag时,缺少Razor语法中的名称空间或程序集,但对于Model来说可以使用

问题描述

|| 我正在使用ASP.NET MVC 3应用程序,该应用程序使用Entity Framework,并且具有一个类库项目,该项目的方法返回“ 0”实例。 我有一个具有以下代码的Controller:
public ActionResult Index()
{
    var students = context.GetStudents();

    return View(students);
}
在这里,GetStudents返回类型为
System.Data.Objects.ObjectSet<Student>
。 在我看来,我有代码“ 3”,可以正确显示模型中的项目数。这很棒。 奇怪的是,如果我改变动作,将ѭ4放入
ViewBag
,那么我就会开始出错。具体来说,如果我这样做:
public ActionResult Index()
{
    var students = context.GetStudents();

    ViewBag.Students = students;

    return View(students);
}
然后在我的视图中添加
@ViewBag.Students.Count()
,我得到一个带有以下消息的死亡黄屏:\“ \'System.Data.Objects.ObjectSet \'不包含\'Count \'\”的定义 为什么它能与模型一起使用(如预期的那样),但不适用于
ViewBag
?     

解决方法

        ViewBag使用新的.NET 4.0动态功能。您的代码由于以下代码无法正常运行的原因完全不起作用:
IEnumerable<int> array = new int[] { 1,2,3 };
dynamic foo = array;
Console.WriteLine(foo.Count());
扩展方法无法在动态对象上解析。您可以通过以下方式使它起作用:
 @(((System.Data.Objects.ObjectSet<Student>)Model).Count())
但请向我保证,您将永远不会做这样的事情。 您的代码无效的事实实际上是一件好事。因为恕我直言,您尝试做的事情在许多方面都是错误的: 您将域模型传递给视图,而不是使用视图模型。 您正在将数据访问特定的对象(例如“ 0”)传递到视图,而不是使用普通的CLR对象。 您正在使用弱类型的ViewBag =>您正在失去Intellisense,您将需要在视图中执行一些强制类型转换,以调用某些辅助方法的某些重载(这将导致意大利面条式代码),由于ViewBag的动态特性,您可以将其命名为... 因此,执行此操作的正确方法是定义一个视图模型,该模型仅包含视图关心的属性。例如:
public class StudentViewModel
{
    public string Name { get; set; }
}
然后您的控制器操作将查询存储库并获取一个模型,该模型将映射到传递给相应的强类型视图的视图模型:
public ActionResult Index()
{
    var model = context.GetStudents();
    var viewModel = model.Select(x => new StudentViewModel
    {
        Name = string.Format(\"{0} {1}\",x.FirstName,x.LastName)
    }).ToArray();
    return View(viewModel);
}
最后,您的强类型视图将仅取决于特定的视图模型(视图不关心数据访问特定的内容,仅应将其传递给需要显示的内容):
@model StudentViewModel[]
<div>We have @Model.Length students</div>
    

相关问答

错误1:Request method ‘DELETE‘ not supported 错误还原:...
错误1:启动docker镜像时报错:Error response from daemon:...
错误1:private field ‘xxx‘ is never assigned 按Alt...
报错如下,通过源不能下载,最后警告pip需升级版本 Requirem...