Maison > développement back-end > Tutoriel C#.Net > [ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

黄舟
Libérer: 2016-12-30 16:00:03
original
1421 Les gens l'ont consulté

[ASP.NET
MVC Maverick Road] 09 - Contrôleur et action (1)

Nous savons que dans MVC, chaque demande sera soumise au contrôleur pour traitement. Le contrôleur est étroitement lié à la requête. Il contient le traitement logique de la requête, peut exploiter le modèle et sélectionner la vue à présenter à l'utilisateur. Le code logique pour les affaires et les données, les interfaces et les bibliothèques de classes auxiliaires ne sont généralement pas placés dans. le Contrôleur.

Controller et Action ont beaucoup de contenu, je l'ai donc divisé en deux parties, ou peut-être trois parties. Cet article présente l'implémentation du Controller, l'acquisition des données d'état par le Controller et le transfert de données d'ActionResult et d'Action. Plus tard, il présentera la Controller Factory, Action Invoker et quelques fonctionnalités plus avancées qui n'ont pas encore été pensées ou. sont en cours d’apprentissage.

Contenu de cet article


Hériter de l'interface IController

Dans les articles précédents de cette série, le contrôleur que nous avons ajouté était hérité de la classe commune abstraite de la classe System.Web.Mvc.Controller (veuillez noter : le contrôleur (ou Controller) et la classe Controller ont deux significations différentes dans cet article, veuillez comprendre en fonction du contexte lors de la lecture de cet article). La classe abstraite Controller encapsule de nombreuses fonctions très pratiques, afin que les développeurs n'aient pas à écrire eux-mêmes des codes de traitement répétitifs et fastidieux.

Si vous n'utilisez pas la classe abstraite Controller encapsulée, nous pouvons également créer votre propre contrôleur en implémentant l'interface IController. Il n'y a qu'une seule méthode Exctute dans l'interface IController :

public interface IController { 
    void Execute(RequestContext requestContext); 
}
Copier après la connexion

L'interface Controller se trouve sous l'espace de noms System.Web.Mvc et est une interface avec une structure très simple.

Lorsqu'une requête est envoyée à une classe de contrôleur qui implémente l'interface IController (le système de routage peut trouver le contrôleur via l'URL demandée), la méthode Execute est appelée.

Ensuite, nous créons une application MVC vide, ajoutons une classe qui implémente IController sous le dossier Controllers et effectuons quelques opérations simples, comme suit :

using System.Web.Mvc;
using System.Web.Routing;

namespace MvcApplication1.Controllers {
    public class BasicController : IController {
        
        public void Execute(RequestContext requestContext) {
            
            string controller = (string)requestContext.RouteData.Values["controller"];
            string action = (string)requestContext.RouteData.Values["action"];
            
            requestContext.HttpContext.Response.Write(
                string.Format("Controller: {0}, Action: {1}", controller, action));
        }
    }
}
Copier après la connexion

Exécutez l'application, l'URL est situé dans /Basic/Index (vous pouvez remplacer Index par n'importe quel autre nom de fragment), le résultat est le suivant :

[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

implémente la classe IController et MVC la reconnaîtra. est une classe de contrôleur, et la requête correspondante est transmise à cette classe pour traitement en fonction du nom du contrôleur.

Mais pour une application un peu complexe, il faut beaucoup de travail pour implémenter soi-même l'interface IController, et nous le faisons rarement. Grâce à cela, nous comprenons mieux le fonctionnement du contrôleur. Tous les traitements de requêtes dans le contrôleur démarrent à partir de la méthode Execute.


Hériter de la classe abstraite Controller

MVC nous permet de la personnaliser et de l'étendre librement. Par exemple, comme mentionné ci-dessus, vous pouvez implémenter l'interface IController pour créer des réponses. diverses demandes. Divers processus et générer des résultats. Si vous n'aimez pas la méthode Action ou si vous ne vous souciez pas de la vue, vous pouvez écrire vous-même un contrôleur meilleur, plus rapide et plus élégant pour gérer la requête. Mais comme mentionné précédemment, implémenter vous-même l’interface IController nécessite beaucoup de travail. Le plus important est que sans tests pratiques à long terme, la robustesse du code ne peut être garantie. La classe System.Web.Mvc.Controller du framework MVC fournit suffisamment de fonctionnalités pratiques pour nous faciliter le traitement des requêtes et le retour des résultats.

Nous avons utilisé à plusieurs reprises le contrôleur qui hérite de la classe Controller et en avons une certaine compréhension. Il offre les fonctionnalités clés suivantes :

Méthode d'action : un contrôleur, dont le comportement est divisé. en plusieurs méthodes. Habituellement, une méthode correspond à une requête, et les données transmises par la requête peuvent être obtenues via les paramètres de la méthode.
ActionResult : Vous pouvez renvoyer un objet qui décrit le résultat de l'exécution de la méthode Action. L'avantage est que vous pouvez simplement spécifier l'objet de retour correspondant si vous souhaitez renvoyer le résultat, et vous n'avez pas à vous inquiéter. sur la façon de l'exécuter et de générer le résultat.
Filtres : grâce aux fonctionnalités C#, le traitement d'un certain comportement (tel que l'autorisation et la vérification) est encapsulé, ce qui facilite la réutilisation entre plusieurs méthodes de contrôleur et d'action.

Donc, si vous n'avez pas de besoins particuliers ou si vous êtes trop occupé, la meilleure façon de créer un Controller qui répond à vos besoins est d'hériter de la classe abstraite Controller. Comme nous l'avons utilisé à de nombreuses reprises auparavant, nous ne ferons pas ici une démonstration détaillée, mais examinons tout de même sa structure de code.

Ajoutez un Controller dans le dossier Controllers. VS a déjà généré la structure de classe pour nous. Si vous aimez la propreté, vous supprimerez habituellement les références inutiles. Le code est le suivant :

.
using System.Web.Mvc;

namespace MvcApplication1.Controllers {
    
    public class DerivedController : Controller {
        
        public ActionResult Index() {
            ViewBag.Message = "Hello from the DerivedController Index method";
            return View("MyView");
        }
    }
}
Copier après la connexion

我们可以查看 Controller 抽象类的定义,发现它是继承 ControllerBase 类的,在 ControllerBase 类中实现了 IController 接口的 Execute 方法,这个方法是MVC对请求进行处理的各个组件的入口,其中包括通过路由系统找到 Action 方法并调用。

Controller 类内部使用 Razor 视图系统来呈现 View,这里通过 View 方法,指定 View 的名称参数来告诉 MVC 选择 MyView 视图来返回给用户结果。


在 Controller 中获取状态数据

我们经常需要访问客户端提交过来的数据,比如 QueryString 值、表单值和通过路由系统来自 URL 的参数值,这些值都可称为状态数据。下面是 Controller 中获取状态数据的三个主要来源:

一系列的上下文对象。
传递给 Action 方法的参数。
显式的调用框架的模型绑定(Model Binding)特性。

从上下文对象中获取状态数据

获取状态数据最直接的方法就是从上下文对象中提取。当你创建了一个继承自 Controller 类的 Controller 时,可以通过一系列的属性可以方便的访问到和请求相关的数据,这些属性包括 Request、Response、RouteData、HttpContext 和 Server,每一个都提供了请求相关的不同类型的信息。下面列出了最常的上下文对象:


[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

在 Action 方法中可以使用任意上下文对象来获取请求相关的信息,如下面在 Action 方法中所演示的:

...
public ActionResult RenameProduct() {
    //访问不同的上下文对象
    string userName = User.Identity.Name;
    string serverName = Server.MachineName;
    string clientIP = Request.UserHostAddress;
    DateTime dateStamp = HttpContext.Timestamp;
    AuditRequest(userName, serverName, clientIP, dateStamp, "Renaming product");
            
    //从POST请求提交的表单中获取数据
    string oldProductName = Request.Form["OldName"];
    string newProductName = Request.Form["NewName"];
    bool result = AttemptProductRename(oldProductName, newProductName);

    ViewData["RenameResult"] = result;
    return View("ProductRenamed");
}
...
Copier après la connexion

这些上下对象不用特意去记,用的时候,你可以通过VS的智能提示来了解这些上下文对象。

使用 Action 方法参数获取状态数据

在本系列的前面的文章中,我们已经知识如何通过 Action 参数来接收数据,这种方法和上面的从上下文对象中获取相比,它更为简洁明了。比如,我们有下面这样一个使用上下文对象的 Action 方法:

public ActionResult ShowWeatherForecast() {
    string city = (string)RouteData.Values["city"];
    DateTime forDate = DateTime.Parse(Request.Form["forDate"]);
    // do something ... 
    return View();
}
Copier après la connexion

我们可以像下面这样使用 Action 方法参数来重写它:

public ActionResult ShowWeatherForecast(string city, DateTime forDate) { 
    // do something ... 
    return View(); 
}
Copier après la connexion

它不仅易读性强,也方便进行单元测试。

Action 方法的参数不允许使用 ref 和 out 参数,这是没有意义的。

MVC 框架通过检查上下文对象来为 Action 方法的参数提供值,它的名称是不区分大小写的,比如 Action 方法的 city 参数的值可以是通过 Request.Form["City"] 来获取的。

理解 Action 方法的参数是如何被赋值的

Controller 类通过 MVC 框架的 value provider 和 model binder 组件来为 Action 方法获取参数的值。

value provider 提供了一系列Controller中可以访问到的值,在内部它通过从 Request.Form、Request.QueryString、Request.Files 和 RouteData.Values 等上下文对象中提取数据(键值集合),然后把数据传递给 model binder,model binder 试图将这些数据与Action方法的参数进行匹配。默认的 model binder 可以创建和赋值给任何.NET类型对象参数(即 Action 方法的参数),包括集合和自定义的类型。

在这不对 model binder 进行介绍,我将在本系列的后续博文中对其进行专门的介绍。


理解 ActionResult

ActionResult 是描述 Action 方法执行结果的对象,它的好处是想返回什么结果就指定对应的返回对象就行,不用关心如何使用Response对象来组织和生成结果。ActionResult 是一个命令模式的例子,这种模式通过存储和传递对象来描述操作。

当 MVC 框架从 Action 方法中接收到一个 ActionResult 对象,它调用这个对象的 ExecuteResult 方法,其内部是通过 Response 对象来返回我们想要的输出结果。

为了更好的理解,我们通过继承 ActionResult 类来自定义一个 ActionResult。在MVC工程中添加一个Infrastructure文件夹,在里面创建一个名为 CustomRedirectResult 的类文件,代码如下:

using System.Web.Mvc;

namespace MvcApplication1.Infrastructure {
    
    public class CustomRedirectResult : ActionResult {
        
        public string Url { get; set; }

        public override void ExecuteResult(ControllerContext context) {
            string fullUrl = UrlHelper.GenerateContentUrl(Url, context.HttpContext);
            context.HttpContext.Response.Redirect(fullUrl);
        }
    }
}
Copier après la connexion

当我们创建一个 CustomRedirectResult 类的实例时,我们可以传递想要跳转的 URL。当 Action 方法执行结束时,MVC 框架调用 ExecuteResult 方法,ExecuteResult 方法通过 ControllerContext 对象获得 Response 对象,然后调用 Redirect 方法。

下面我们在 Controller 中使用自定义的 CustomRedirectResult:

public class DerivedController : Controller {
    ...
    public ActionResult ProduceOutput() {
        if (Server.MachineName == "WL-PC") {
            return new CustomRedirectResult { Url = "/Basic/Index" };
        }
        else {
            Response.Write("Controller: Derived, Action: ProduceOutput");
            return null;
        }
    }
Copier après la connexion

运行后我们看到如下结果:

[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

当运行在本机(WL-PC)时直接重定向到了指定的/Basic/Index。

上面我们通过自定义 CustomRedirectResult 来实现重定向,我们可以用 MVC 框架提供的方法,如下:

... 
public ActionResult ProduceOutput() { 
    return new RedirectResult("/Basic/Index"); 
}
Copier après la connexion

为了使用方便,Controller 类中为大部分类型的 ActionResult 提供简便的方法,如上面的可像下面这样简写:

... 
public ActionResult ProduceOutput() { 
    return Redirect("/Basic/Index"); 
}
Copier après la connexion

MVC框架包含了许多 ActionResult 类型,这些类型都继承自 ActionResult 类,大部分在 Controller 类中都有简便的方法,下面列举了一些:

[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

除了该表列出来的,还有ContentResult、FileResult、JsonResult 和 JavaScriptResult。具体每种ActionResult类型的用法这里就不讲了,大家可以看看蒋老师的了解ASP.NET
MVC几种ActionResult的本质系列的文章。


几种从 Action 传递数据到 View 的方式

我们经常需要在 Action 方法中传递数据到一个 View 中,MVC 框架为此提供了一些很方便的操作。下面简单简介几种常用的方式。

View Model 对象

通过 View Model 对象传递数据给View,这是最常用的一种,在 Acton 方法执行结束时通过 View 方法传递 View Model 对象给 View,如下代码所示:

... 
public ViewResult Index() { 
    DateTime date = DateTime.Now; 
    return View(date); 
}
Copier après la connexion

在 View 中我们通过 Model 属性来使用传递过来的 View Model 对象,如下:

@model DateTime 

@{ 
    ViewBag.Title = "Index"; 
}

<h2>Index</h2> 
The day is: @Model.DayOfWeek
Copier après la connexion

在 Razor 视图引擎中,@model 的作用是声明 odel 属性的类型,省去了类型转换的麻烦,而 @Model 是V iew Model 对象的引用。

ViewBag、ViewData 和 TempData 属性

ViewBag、ViewData 和 TempData 都是 Controller 和 View 中能访问到的属性,都是用来存储小量的数据,他们的区别如下:

ViewBag,是一个动态(dynamic)的弱类型,在程序运行的时候解析,是 MVC3 中新增的特性,只在当前View有效。
ViewData,是一个字典集合,也是只在当前View有效,性能比 ViewBag 高,但是使用的时候需要类型转换。
TempData,也是字典集合,一般用于两个请求之间临时缓存内容或页面间传递消息,保存在 Session 中,使用完以后则从 Session 中被清除。

下面是三者使用的例子,先在 Controller 中分别用三者存储小数据:

public class DerivedController : Controller {

    public ActionResult Index() {
        ViewBag.DayOfWeek = DateTime.Now.DayOfWeek;
        ViewData["DayOfMonth"] = DateTime.Now.Day;
        return View();
    }

    public ActionResult ProduceOutput() {
        TempData["Message"] = "Warning message from Derived Controller.";
        return Redirect("/Home/Index");
    }
}
Copier après la connexion

在 Views/Derived 目录下的 Index.cshtml 中,取出 ViewBag 和 ViewData 中的存储的数据:

...
Day of week from ViewBag: @ViewBag.DayOfWeek
<p /> 
Day of month from ViewData: @ViewData["DayOfMonth"]
Copier après la connexion

在 Views/Home 目录下的 Index.cshtml 中,取 TempData 中的数据如下:

...
@TempData["Message"]
Copier après la connexion

当请求 /Derived/ProduceOutput 时,ProduceOutput 方法将一条消息存到 TempData 中,并跳转到 /Home/Index。

下面是分别是将URL定位到 /Derived/Index 和 /Derived/ProduceOutput 时的结果:

[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

[ASP.NET MVC Mavericks Road] 09 - Contrôleur et action (1)

一般在当前 View 中使用 ViewBag 或 ViewData,在两个请求之间传递临时数据用 TempData。由于 TempData 被使用后即被释放,所以如果要二次使用 TempData 中的数据就需要将其存到其他变量中。

 以上就是[ASP.NET MVC 小牛之路]09 - Controller 和 Action (1)的内容,更多相关内容请关注PHP中文网(www.php.cn)!


Étiquettes associées:
source:php.cn
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal