ASP.NET MVC中Controller控制器向View视图传值的几种方式

2022-04-17 15:15:04

一、准备工作

创建一个ASP.NET MVC程序,然后在Models文件夹里面新添加Student实体类,用来模拟从Controller向View传递数据,Student类定义如下:

using System;using System.Collections.Generic;using System.Linq;using System.Web;namespace MVCStudyDemo.Models{    public class Student    {        public int ID { get; set; }        public string Name { get; set; }        public int Age { get; set; }        public string Sex { get; set; }        public string Email { get; set; }    }}

二、通过ViewData传值

MVC从开始版本就一直支持使用ViewData将Controller里面的数据传递到View。ViewData定义如下:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

从上面的截图中可以看出,ViewData里面存的是字典类型的数据,在查看ViewDataDictionary的定义:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

"Data"] = new List<Student>() { new Student { ID = 1, Name = "唐僧", Age = 34, Sex = "男", Email = "747976523@qq.com" }, new Student { ID = 2, Name = "孙悟空", Age = 635, Sex = "男", Email = "sunwukong@163.com" }, new Student { ID = 3, Name = "白骨精", Age = 4532, Sex = "女", Email = "74345523@qq.com" } }; // 通过ViewBag传值 ViewBag.Name = "ViewBag"; ViewBag.StudentData = new Student() { ID = 5, Name = "沙悟净", Age = 567, Sex = "男", Email = "4567890345@qq.com" }; // 返回同名视图 return View(); } }}

对应的Index视图代码:

@*引入Student的命名空间*@@using MVCStudyDemo.Models;@{    ViewBag.Title = "Index";    // ViewBag是dynamic类型的,使用的时候不需要进行类型转换    var stu = ViewBag.StudentData;    var stuList = ViewBag.Data;}<h2>通过ViewBag向View传递数据</h2><div class="jumbotron">    <div>        <div>            Controller通过ViewBag向View传递数据        </div>        <div>            1、通过ViewData传递字符串 ViewData["name"]:@ViewData["name"];        </div>        <div>            2、通过ViewBag传递字符串 ViewBag.name:@ViewBag.Name;        </div>        <div>            3、输出stu            <div>                ID:@stu.IDName:@stu.NameAge:@stu.AgeSex:@stu.SexEmail:@stu.Email            </div>            4、输出stuList            @foreach (var item in stuList)            {                <div>                    ID:@item.IDName:@item.NameAge:@item.AgeSex:@item.SexEmail:@item.Email                </div>            }        </div>    </div></div>

运行结果;

ASP.NET MVC中Controller控制器向View视图传值的几种方式

看了上面的运行结果,你可能会提出如下的两个疑问:

1、Controller里面没有定义ViewBag.Data,为什么在这里可以使用呢?

这是因为ViewBag是从MVC3版本才开始出现的,为了兼容以前的ViewData,所以这里虽然没有定义ViewBag.Student,但是ViewBag可以使用ViewData里面定义的Data数据。

2、ViewData["name"]和ViewBag.name的值是一样的

在控制器里面明明设置的两个值是不同的,但是为什么这里都变成一样的了呢?这是因为ViewData和ViewBag的属性是重叠的,两者都是字典类型的,一切以后面定义的属性为准,即后面定义的会覆盖前面定义的。

修改Controller代码,将ViewData的顺序放到ViewBag后面:

using MVCStudyDemo.Models;using System;using System.Collections.Generic;using System.Linq;using System.Web;using System.Web.Mvc;namespace MVCStudyDemo.Controllers{    public class ViewBagDemoController : Controller    {        // GET: ViewBagDemo        /// <summary>        /// 通过ViewBag向View传递数据        /// </summary>        /// <returns></returns>        public ActionResult Index()        {            // 通过ViewData传值            //ViewData["name"] = "ViewData";            // 传递集合到View            ViewData["Data"] = new List<Student>()            {                new Student               {                 ID = 1,                 Name = "唐僧",                 Age = 34,                 Sex = "男",                 Email = "747976523@qq.com"               },               new Student               {                 ID = 2,                 Name = "孙悟空",                 Age = 635,                 Sex = "男",                 Email = "sunwukong@163.com"               },               new Student               {                 ID = 3,                 Name = "白骨精",                 Age = 4532,                 Sex = "女",                 Email = "74345523@qq.com"               }            };            // 通过ViewBag传值            ViewBag.Name = "ViewBag";            ViewBag.StudentData = new Student()            {                ID = 5,                Name = "沙悟净",                Age = 567,                Sex = "男",                Email = "4567890345@qq.xAhDFvCXCmcom"            };            // 把ViewData的顺序放到ViewBag后面            ViewData["name"] = "ViewData";            // 返回同名视图            return View();        }    }}

在查看Index视图显示效果:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

这时会发现,Index视图里面显示的都是ViewData的值了。

四、通过TempData传值

查看TempData的定义:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

我们发现TempData也是字典类型的。

新建Controller,并命名为TempDataDemo,该Controller用来模拟通过TempData向View传递数据。

ASP.NET MVC中Controller控制器向View视图传值的几种方式

Controller代码:

using MVCStudyDemo.Models;using System;using System.Collections.Generic;using System.Linq;using System.Web;using System.Web.Mvc;namespace MVCStudyDemo.Controllers{    public class TempDataDemoController : Controller    {        // GET: TempDataDemo        public ActionResult Index()        {            // ViewData            ViewData["Name"] = "tom";            // ViewBage            ViewBag.Name = "Jon";            // TempData            TempData["Name"] = "Andi";            TempData["Stu"] = new Student()            {                ID = 5,                Name = "沙悟净",                Age = 567,                Sex = "男",                Email = "4567890345@qq.com"            };            // 返回同名视图            return View();        }    }}

对应的Index视图代码,TempData也是字典类型的,所以在View页面中使用的时候也需要进行类型转换:

@*引入Student的命名空间*@@using MVCStudyDemo.Models;@{    ViewBag.Title = "Index";}@{    // 类型转换    var stu = TempData["Stu"] as Student;}<h2>通过TempData向View传递数据</h2><div class="jumbotron">    <p>        <div>            <div>                通过ViewData传递字符串 ViewData["Name"]:@ViewData["Name"];            </div>        </div>    </p>    <p>        <div>            <div>                通过ViewBag传递字符串 ViewBag.Name:@ViewBag.Name;            </div>        </div>    </p>    <p>        <div>            <div>                1、传递字符串 TempData["Name"]:@TempData["Name"];            </div>            2、输出stu            <div>                ID:@stu.IDName:@stu.NameAge:@stu.AgeSex:@stu.SexEmail:@stu.Email            </div>        </div>    </p></div>

运行结果:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

从上面的结果中可以看出:TempData的属性值不会覆盖上面定义的属性值。那TempData还有什么作用呢?在看下面的代码:

using MVCStudyDemo.Models;using System;using System.Collections.Generic;using System.Linq;using System.Web;using System.Web.Mvc;namespace MVCStudyDemo.Controllers{    public class TempDataDemoController : Controller    {        // GET: TempDataDemo        public ActionResult Index(int? id)        {            // ViewData            ViewData["Name"] = "tom";            // ViewBage            ViewBag.Name = "Jon";            // TempData            TempData["Name"] = "Andi";            TempData["Stu"] = new Student()            {                ID = 5,                Name = "沙悟净",                Age = 567,                Sex = "男",                Email = "4567890345@qq.com"            };            if(id==null)            {                // 跳转到TempDataTest方法                return RedirectToAction("TempDataTest");            }            else            {                // 返回同名视图                return View();            }                  }        public ActionResult TempDataTest()        {            // 返回同名视图            return View();        }    }}

TempDataTest视图代码:

@*引入Student的命名空间*@@using MVCStudyDemo.Models;@{    ViewBag.Title = "TempDataTest";    // 类型转换    var stu = TempData["Stu"] as Student;}<h2>TempDataTest</h2><h3>ViewData["Name"]:@ViewData["Name"]</h3><h3>ViewBage.Name:@ViewBag.Name</h3><h3>TempData["Name"]:@TempData["Name"];</h3><h3>ID:@stu.IDName:@stu.NameAge:@stu.AgeSex:@stu.SexEmail:@stu.Email</h3>

我们先在浏览器里面输入下面的地址:http://localhost:1098/TempDataDemo/index/12,这里给id传值12,根据代码,会显示Index视图:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

我们看到:ViewData、ViewBag和TempData都可以取到值。

在浏览器里面输入下面的URL地址:http://localhost:9080/TempDataDemo/index,这就表示传递的id是null值,根据代码,会返回TempDataTest视图,运行结果:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

从结果会发现这时ViewData和ViewBag都取不到数据了,只有TempData可以取到数据,可以得出TempData和ViewData、ViewBag的区别:

ViewData和ViewBag不能跨Action方法使用,TempData可以跨Action方法使用,因为TempData是基于session存储的。TempData只能跨Action方法访问一次,再次访问数据也会丢失。

五、Model传值

在Action放过里面的View上面F12转到定义:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

可以看到View有很多重载,其中一种可以直接传递model,如上面截图中红框所示。

新建Controller,并命名为ModelDemo,该Controller用来模拟通过Model向View传递数据。

ASP.NET MVC中Controller控制器向View视图传值的几种方式

对应的controller代码:

using MVCStudyDemo.Models;using System;using System.Collections.Generic;using System.Linq;using System.Web;using System.Web.Mvc;namespace MVCStudyDemo.Controllers{    public class ModelDemoController : Controller    {        // GET: ModelDemo        public ActionResult Index()        {            Student student = new Student()            {                ID = 5,                Name = "沙悟净",                Age = 567,                Sex = "男",                Email = "4567890345@qq.com"            };            // 返回model            return View(student);            // 或者直接返回            //return View(new Student()            //{            //    ID = 5,            //    Name = "沙悟净",            //    Age = 567,            //    Sex = "男",            //    Email = "4567890345@qq.com"            //});        }    }}

Index视图代码:

@{    ViewBag.Title = "Index";}@*不需要进行类型转换*@<h2>通过Model向View传值</h2><h3>ID:@Model.ID</h3><h3>Name:@Model.Name</h3><h3>Age:@Model.Age</h3><h3>Sex:@Model.Sex</h3><h3>Email:@Model.Email</h3>

运行结果:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

六、总结

最后总结一下ViewData和ViewBag的区别:

ASP.NET MVC中Controller控制器向View视图传值的几种方式

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。