在下面的RedirectToAction
中,我想传递一个viewmodel
。如何将模型传递给重定向?
我设置了一个断点来检查模型的值,以验证模型是否正确创建。它是正确的,但结果视图不包含模型属性中找到的值。
//
// model created up here...
//
return RedirectToAction("actionName", "controllerName", model);
ASP.NET MVC 4 RC
答案 0 :(得分:88)
RedirectToAction
向客户端浏览器返回302响应,因此浏览器会向响应的位置标头值中的url发出新的GET请求来到浏览器。
如果您尝试将简单的平面视图模型传递给第二个操作方法,则可以使用RedirectToAction
方法的this overload。
protected internal RedirectToRouteResult RedirectToAction(
string actionName,
string controllerName,
object routeValues
)
RedirectToAction
会将传递的对象(routeValues)转换为查询字符串并将其附加到url(从我们传递的前2个参数生成),并将生成的url嵌入location回复的标题。
我们假设您的视图模型是这样的
public class StoreVm
{
public int StoreId { get; set; }
public string Name { get; set; }
public string Code { set; get; }
}
在第一个操作方法中,您可以将此对象传递给RedirectToAction
方法,如下所示
var m = new Store { StoreId =101, Name = "Kroger", Code = "KRO"};
return RedirectToAction("Details","Store", m);
此代码将向浏览器发送302响应,其中位置标头值为
Store/Details?StoreId=101&Name=Kroger&Code=KRO
假设您的Details
操作方法的参数类型为StoreVm
,则查询字符串参数值将正确映射到参数的属性。
public ActionResult Details(StoreVm model)
{
// model.Name & model.Id will have values mapped from the request querystring
// to do : Return something.
}
以上将适用于传递小平视图模型。但是如果你想传递一个复杂的对象,你应该尝试遵循PRG模式。
PRG代表 POST - REDIRECT - GET 。使用此方法,您将在查询字符串中发出具有唯一ID的重定向响应,使用该响应,第二个GET操作方法可以再次查询资源并将某些内容返回到视图。
int newStoreId=101;
return RedirectToAction("Details", "Store", new { storeId=newStoreId} );
这将创建网址Store/Details?storeId=101
在您的详细信息GET
操作中,使用传入的storeId,您将从某个地方(从服务或查询数据库等)获取/构建StoreVm
对象
public ActionResult Details(string storeId)
{
// from the storeId value, get the entity/object/resource
var store = yourRepo.GetStore(storeId);
if(store!=null)
{
// Map the the view model
var storeVm = new StoreVm { Id=storeId, Name=store.Name,Code=store.Code};
return View(storeVm);
}
return View("StoreNotFound"); // view to render when we get invalid store id
}
遵循PRG pattern是处理此用例的更好解决方案。但是,如果您不想这样做并且确实希望在无状态HTTP 请求中传递一些复杂数据,则可以使用一些临时存储机制,如TempData
TempData["NewCustomer"] = model;
return RedirectToAction("Index", "Users");
再次在GET
动作方法中阅读。
public ActionResult Index()
{
var model=TempData["NewCustomer"] as Customer
return View(model);
}
TempData
使用场景后面的Session
对象来存储数据。但是一旦读取数据,数据就会终止。
Rachel写了一篇很好的博客post,解释何时使用TempData / ViewData。值得一读。
在Asp.Net核心中,无法在TempData中传递复杂类型。您可以传递简单类型,例如string
,int
,Guid
等。
如果您绝对想通过TempData传递复杂类型对象,则有2个选项。
1)将对象序列化为字符串并传递它。
以下是使用Json.NET将对象序列化为字符串
的示例var s = Newtonsoft.Json.JsonConvert.SerializeObject(createUserVm);
TempData["newuser"] = s;
return RedirectToAction("Index", "Users");
现在,在Index
操作方法中,从TempData中读取此值并将其反序列化为CreateUserViewModel
类对象。
public IActionResult Index()
{
if (TempData["newuser"] is string s)
{
var newUser = JsonConvert.DeserializeObject<CreateUserViewModel>(s);
// use newUser object now as needed
}
// to do : return something
}
2)将简单类型的字典设置为TempData
var d = new Dictionary<string, string>
{
["FullName"] = rvm.FullName,
["Email"] = rvm.Email;
};
TempData["MyModelDict"] = d;
return RedirectToAction("Index", "Users");
稍后阅读
public IActionResult Index()
{
if (TempData["MyModelDict"] is Dictionary<string,string> dict)
{
var name = dict["Name"];
var email = dict["Email"];
}
// to do : return something
}
答案 1 :(得分:0)
我总是喜欢添加属性
public string ActionName { get; set; }
到我的ViewModel(通常也到我的模型)。从那里我使用
`@Html.Action(@ Model.ActionName,someModelRouteValues)
我实际上将“ ActionName”存储在db实体中。这样就与视图无关。这与数据库实体模型中的数据有关(可以轻松操作)。
答案 2 :(得分:0)
另一种方法是将其存储在会话中。
var s = JsonConvert.SerializeObject(myView);
HttpContext.Session.SetString("myView", s);
并取回它
string s = HttpContext.Session.GetString("myView");
myView = JsonConvert.DeserializeObject<MyView>(s);