希望实现的效果是:对购物车中所有商品的总价,实现9折或8折:

当点击"9折":

当点击"8折":

□ 思路

8折或9折是打折接口的不同实现,关键是:由什么条件决定使用哪种打折方式?

--当点击8折或9折链接的时候,把参数放在路由中,然后在自定义控制器工厂中根据参数的不同选择使用哪种打折方式。

□ model

public class CartLine
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public decimal Price { get; set; }
        public int Quantity { get; set; }
    }

□ 接口

using MvcApplication2.Models;

namespace MvcApplication2
{
    public interface IDiscount
    {
        decimal GetFinalPrice(List<CartLine> cartLines);
    }
}

□ 接口的2种实现

using System.Collections.Generic;
 
namespace MvcApplication2.implementation
{
    public class NineDiscount : IDiscount
    {
        public decimal GetFinalPrice(List<Models.CartLine> cartLines)
        {
            decimal result = 0.0M;
            foreach (var item in cartLines)
            {
                result += item.Price*item.Quantity;
            }
            return result*(90M/100M);
        }
    }
}
 
 
 
using System.Collections.Generic;
 
namespace MvcApplication2.implementation
{
    public class EightDiscount : IDiscount
    {
        public decimal GetFinalPrice(List<Models.CartLine> cartLines)
        {
            decimal result = 0.0M;
            foreach (var item in cartLines)
            {
                result += item.Price * item.Quantity;
            }
            return result * (80M / 100M);
        }
    }
}
 

□ HomeController

using System.Collections.Generic;
using System.Web.Mvc;
using MvcApplication2.Models;
 
namespace MvcApplication2.Controllers
{
    public class HomeController : Controller
    {
        public ActionResult Index()
        {
            List<CartLine> cartLines = new List<CartLine>()
            {
                new CartLine(){Id = 1, Name = "Product1", Price = 80M, Quantity = 2},
                new CartLine(){Id = 2, Name = "Product2", Price = 100M, Quantity = 3},
            };
            Session["cart"] = cartLines;
            return View(cartLines);
        }
 
    }
}
 

□ Home/Index.cshtml

把不同的打折方式放在路由中传递。

@model List<MvcApplication2.Models.CartLine>
 
@{
    ViewBag.Title = "Index";
    Layout = "~/Views/Shared/_Layout.cshtml";
}
 
<h2>Index</h2>
 
<table>
    <tr style="background-color: #e3e3e3;">
        <td>产品</td>
        <td>价格</td>
        <td>数量</td>
    </tr>
    @foreach (var item in Model)
    {
        <tr>
            <td>@item.Name</td>
            <td>@string.Format("{0:C}", item.Price)</td>
            <td>@item.Quantity</td>
        </tr>
    }
</table>
 
<p>
    @Html.ActionLink("9折购买", "Index", "Shop", new {policy = "Nine"},new {})
</p>
 
<p>
    @Html.ActionLink("8折购买", "Index", "Shop", new {policy = "Eight"},new {})
</p>    
 

□ 自定义控制器工厂,使用Ninject,根据路由参数policy的不同,决定选择具体的打折接口实现

using System;
using System.Web.Mvc;
using System.Web.Routing;
using MvcApplication2.implementation;
using Ninject;
 
namespace MvcApplication2.Extension
{
    public class NinjectControllerFactory : DefaultControllerFactory
    {
        IKernel ninjectKernel;
        string policy = "";
 
        public NinjectControllerFactory()
        {
            ninjectKernel = new StandardKernel();
        }
 
        protected override IController GetControllerInstance(RequestContext requestContext, Type controllerType)
        {
            if (requestContext.RouteData.Values["policy"] != null)
            {
                policy = requestContext.RouteData.Values["policy"].ToString();
            }
            AddBindings();
            return controllerType == null ? null : (IController) ninjectKernel.Get(controllerType);
        }
 
        private void AddBindings()
        {
            switch (policy)
            {
                case "Eight":
                    ninjectKernel.Rebind<IDiscount>().To<EightDiscount>();
                    break;
                case "Nine":
                    ninjectKernel.Rebind<IDiscount>().To<NineDiscount>();
                    break;
                default:
                    ninjectKernel.Rebind<IDiscount>().To<NineDiscount>();
                    break;
            }
        }
    }
}
 

□ 自定义控制器工厂全局注册

ControllerBuilder.Current.SetControllerFactory(new NinjectControllerFactory());

□ ShopController中使用打折接口方法

using System;
using System.Collections.Generic;
using System.Web.Mvc;
using MvcApplication2.Models;
 
 
namespace MvcApplication2.Controllers
{
    public class ShopController : Controller
    {
        public IDiscount _Discount;
 
        public ShopController(IDiscount discount)
        {
            this._Discount = discount;
        }
 
        public ActionResult Index(string policy)
        {
            List<CartLine> cartLines = new List<CartLine>();
            if (Session["cart"] != null)
            {
                cartLines = (List<CartLine>)Session["cart"];
            }
            ViewData["total"] = String.Format("{0:C}",_Discount.GetFinalPrice(cartLines));
            return View();
        }
 
    }
}
 

□ Shop/Index.cshtml

@{
    ViewBag.Title = "Index";
    Layout = "~/Views/Shared/_Layout.cshtml";
}
 
打折后的价格为: @ViewData["total"]

□ 自定义路由

为了让url更直观,符合controller/action/paramter:

            routes.MapRoute(
                name: "Default",
                url: "{controller}/{action}/{policy}",
                defaults: new { controller = "Home", action = "Index", policy = UrlParameter.Optional }
            );

MVC扩展控制器工厂,通过继承DefaultControllerFactory来决定使用哪个接口实现,使用Ninject的更多相关文章

  1. MVC扩展控制器工厂,通过实现IControllerFactory,根据action名称生成不同的Controller

    关于控制器工厂的扩展,要么通过实现IControllerFactory接口,要么通过继承DefaultControllerFactory.本篇中,我想体验的是: 1.当请求经过路由,controlle ...

  2. MVC扩展控制器, 把部分视图转换成字符串(带验证信息), 并以json传递给前端视图

    当我们使用jQuery异步提交表单数据的时候,需要把部分视图转换成字符串(带验证信息),以json的形式传递给前端视图.   使用jQuery异步加载部分视图,返回内容追加到页面某个div:   jQ ...

  3. MVC项目实践,在三层架构下实现SportsStore-03,Ninject控制器工厂等

    SportsStore是<精通ASP.NET MVC3框架(第三版)>中演示的MVC项目,在该项目中涵盖了MVC的众多方面,包括:使用DI容器.URL优化.导航.分页.购物车.订单.产品管 ...

  4. .NET/ASP.NET MVC Controller 控制器(IController控制器的创建过程)

    阅读目录: 1.开篇介绍 2.ASP.NETMVC IControllerFactory 控制器工厂接口 3.ASP.NETMVC DefaultControllerFactory 默认控制器工厂 4 ...

  5. .NET/ASP.NET MVC Controller 控制器(深入解析控制器运行原理)

    阅读目录: 1.开篇介绍 2.ASP.NETMVC Controller 控制器的入口(Controller的执行流程) 3.ASP.NETMVC Controller 控制器的入口(Controll ...

  6. 三、ASP.NET MVC Controller 控制器(二:IController控制器的创建过程)

    阅读目录: 1.开篇介绍 2.ASP.NETMVC IControllerFactory 控制器工厂接口 3.ASP.NETMVC DefaultControllerFactory 默认控制器工厂 4 ...

  7. 二、ASP.NET MVC Controller 控制器(一:深入解析控制器运行原理)

    阅读目录: 1.开篇介绍 2.ASP.NETMVC Controller 控制器的入口(Controller的执行流程) 3.ASP.NETMVC Controller 控制器的入口(Controll ...

  8. NET/ASP.NET MVC Controller 控制器(一:深入解析控制器运行原理)

    阅读目录: 1.开篇介绍 2.ASP.NETMVC Controller 控制器的入口(Controller的执行流程) 3.ASP.NETMVC Controller 控制器的入口(Controll ...

  9. ASP.NET MVC 创建控制器类过程

    MvcHandler.ProcessRequestInit()方法: 1.1获取控制器的名称string requiredString = this.RequestContext.RouteData. ...

随机推荐

  1. IntelliJ IDEA 修改IDE字体、代码字体。

    IntelliJ IDEA 默认的 IDE 菜单字体太小,看着不舒服 ,我们调节下: ==============以上修改 仅仅针对的IDE字体,对代码的字体不生效. 所以如果代码 你觉得小 还得修改 ...

  2. Centos之压缩和解压缩命令

    常用压缩格式:.zip .gz .bz2 常用压缩格式:.tar.gz  .tar.bz2 zip格式压缩 zip压缩文件名 源文件 压缩文件 zip -r 压缩文件名 源目录 压缩目录 [root@ ...

  3. Linux 相关

    一.WCHAN的含义 WCHAN 进程正在睡眠的内核函数名称:该函数的名称是从/root/system.map文件中获得的. 参考:解析ANDROID ps命令执行后各项参数的含义 二.查看线程 ps ...

  4. Tensorflow之训练MNIST(1)

    先说我遇到的一个坑,在下载MNIST训练数据的时候,代码报错: urllib.error.URLError: <urlopen error [SSL: CERTIFICATE_VERIFY_FA ...

  5. vue.js学习 自定义过滤器使用(1)

    在这个教程中,我们将会通过几个例子,了解和学习VueJs的过滤器.我们参考了一些比较完善的过滤器,比如orderBy 和 filterBy.而且我们可以链式调用过滤器,一个接一个过滤.因此,我们可以定 ...

  6. xss攻击原理与解决方法

    概述 XSS攻击是Web攻击中最常见的攻击方法之一,它是通过对网页注入可执行代码且成功地被浏览器 执行,达到攻击的目的,形成了一次有效XSS攻击,一旦攻击成功,它可以获取用户的联系人列 表,然后向联系 ...

  7. CCF CSP 201609-4 交通规划

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201609-4 交通规划 问题描述 G国国王来中国参观后,被中国的高速铁路深深的震撼,决定为自 ...

  8. C语言:1孩半问题

    题目: 一孩半,又称独女户二胎,即中国大陆部分农村的一项计划生育政策,第一胎是女孩的夫妻可以生育第二个子女.如果第二胎有n%人工性别选择干预(选择男孩),试问男女比例为多少.(10分)题目内容: 一孩 ...

  9. 【LOJ】#2068. 「SDOI2016」探险路线

    题解 少考虑了情况,导致我以为是暴力讨论一次角落移动 de了两天才反应过来--简直降智 事实上,我们把移动分三类,一种是在边界跳过一段,一种是在左上角上左上左上左这样撞墙,在右下角下右下右下右这么撞墙 ...

  10. react篇章-React Props-Props 验证

    React.PropTypes 在 React v15.5 版本后已经移到了 prop-types 库. <script src="https://cdn.bootcss.com/pr ...