C# MVC模式中应该怎样区分应用程序逻辑(Controller层)和业务逻辑(Model层)?

 更新时间:2015年06月16日 10:53:10   投稿:junjie  
这篇文章主要介绍了C# MVC模式中应该怎样区分应用程序逻辑(Controller层)和业务逻辑(Model层)?,这也小编做.NET项目时经常思考和让人混乱的一个问题,这篇文章写的挺好,一下清晰了许多,需要的朋友可以参考下

现在的大部分框架都是 MVC 模式,但 MVC 三个部分怎么配合,这里做了一点总结:

基本原则:业务逻辑代码应该写在 M 里面,而应用程序逻辑应该写在 C 里面。V 只是单纯的展示数据。

举个简单例子吧:用户往购物车添加一个商品

用户点击商品的“添加到购物车”按钮,引起一次请求。服务器开始处理该请求,过程:

1、检查当前用户是否有权限(比如是否已经登录、用户帐户状态、是否可以购物等)
2、检查要添加的商品ID是否有效、
3、检查要添加的商品库存是否足够
4、将商品加入购物车,并保存购物车状态
5、反馈信息

在上述流程中:

1: 是应用程序逻辑(一般由框架实现):因为和“添加商品到购物车”这个业务没有直接关系
2: 业务逻辑:不能购买不存在的商品,这是业务进行的基本条件
3: 业务逻辑:商品库存决定了是否可以购买此商品,这是业务进行的基本条件
4: 业务逻辑
5: 应用程序逻辑

用代码表示的,可能像下面这样:

复制代码 代码如下:

// Cart控制器
class Controller_Cart
{
    function actionAddGoods()
    {
        $goods_id = (int)$_GET['goods_id'];
        Cart::instance()->add($goods_id)->save();
       
        echo '添加成功';
    }
}


// Cart 模型
class Cart
{
    /**
     * 购物车中的所有项目
     */
    public $items = array();
  
    /**
     * 单子模式,返回购物车对象的唯一实例
     */
    static function instance()
    {
        ...
    }
  
    function add($goods_id, $quantity = 1)
    {
        $goods = Goods::find($goods_id)->get();
        // 检查 id 和库存数
        if ($goods->id && $quantity > $goods->remaining)
        {
            // 添加商品到购物车
            $this->items[] = array($goods, $quantity);
        }
        else
        {
            throw new CartExecption('无效的商品 ID');
        }
        return $this;
    }
}


这个代码不完整,但是演示了最重要的部分,就是应用程序逻辑和业务逻辑的分离。
如果这个流程走下去,用户要结算了,那么代码如下:
复制代码 代码如下:

class Controller_Cart
{
    function actionCheckOut()
    {
        Cart::instance()->checkout();
       
        echo '成功';
    }
}
class Cart
{
    function checkout()
    {
        // 开启一个数据库事务
        ....
       
        try
        {
            // 创建一个新的订单对象
            // $this->owner 是当前购物车的所有者(用户)
            $order = new Order($this->owner);
           
            // 将购物车中的所有商品添加到订单中
            foreach ($this->items as $item)
            {
                list($goods, $quantity) = $item;
                $order->add($goods, $quantity);
            }
            // 保存订单
            $order->save();
           
            // 清空购物车
            $this->items = array();
        }
        catch (Exception $ex)
        {
            // 出错了,回滚事务
            ....
              
            // 再重新抛出异常
            throw $ex;
        }
       
        // 返回新建的订单
        return $order;
    }
}


class Order extends Model
{
    public $items;
  
    function add($goods, $quantity)
    {
        $this->items[] = array($goods, $quantity);
        return $this;
    }
  
    function save()
    {
        foreach ($this->items as $item)
        {
            list($goods, $quantity) = $item;
            // 保存订单时,减少订单中每一个商品的库存数
            $goods->decrRemaining($quantity);
        }
       
        // 调用父类的保存
        parent::save();
       
        return $this;
    }
}


结算的代码很容易理解:
1、调用购物车的 checkout() 方法
2、开启数据库事务,这样当保存订单失败时(例如库存数不够)则回滚,确保数据库内容没有受影响
3、将购物车中的所有商品添加到订单
4、调用订单对象的 save() 方法
  4.1、遍历订单的所有项目,减少商品的库存(如果此时失败,商品的 decrRemaining() 方法会抛出异常)
  4.2、调用模型父类的 save() 方法
5、清空购物车,返回新建的订单对象

整个流程我们假定创建订单就等同于客户确认订单,此时减少库存。也有可能是后台确认订单配货后才减少库存,这和卖家的经营策略有关。

这两个例子里面,业务逻辑都在模型中实现,控制器(也就是封装应用程序逻辑的层)仅仅完成处理输入数据、调用业务方法、反馈结果等任务。

相关文章

  • 解析如何正确使用SqlConnection的实现方法

    解析如何正确使用SqlConnection的实现方法

    本篇文章对如何正确使用SqlConnection的实现方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C#实现文件上传及文件下载功能实例代码

    C#实现文件上传及文件下载功能实例代码

    文件上传文件下载需求在项目中经常会遇到,今天小编给大家分享C#实现文件上传及文件下载功能实例代码,需要的朋友参考下吧
    2017-08-08
  • c#使用Dataset读取XML文件动态生成菜单的方法

    c#使用Dataset读取XML文件动态生成菜单的方法

    这篇文章主要介绍了c#使用Dataset读取XML文件动态生成菜单的方法,涉及C#使用Dataset操作XML文件的相关技巧,需要的朋友可以参考下
    2015-05-05
  • C#中的除法运算符与VB.NET中的除法运算符

    C#中的除法运算符与VB.NET中的除法运算符

    这篇文章主要介绍了C#中的除法运算符与VB.NET中的除法运算符,需要的朋友可以参考下
    2014-10-10
  • C#执行js动态编译的方法

    C#执行js动态编译的方法

    这篇文章主要介绍了C#执行js动态编译的方法,是涉及动态编译脚本非常实用的技巧,需要的朋友可以参考下
    2015-01-01
  • 详解Unity安卓共享纹理

    详解Unity安卓共享纹理

    本文主要介绍了Unity安卓共享纹理,对此感兴趣的同学,可以参考下,亲自实验一下,理解其原理。
    2021-05-05
  • C#编写控制台程序纸牌游戏

    C#编写控制台程序纸牌游戏

    这篇文章主要为大家详细介绍了C#编写控制台程序纸牌游戏,适合C#初学者,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-08-08
  • C#9.0 新特性简介

    C#9.0 新特性简介

    这篇文章主要介绍了C#9.0 新特性简介,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-10-10
  • C#中new的几种用法详解

    C#中new的几种用法详解

    本文主要介绍了C#中new的几种用法,具有很好的参考价值,下面跟着小编一起来看下吧
    2017-02-02
  • 详解c# 中的DateTime

    详解c# 中的DateTime

    这篇文章主要介绍了c# 中的DateTime的相关资料,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-07-07

最新评论