背景

系统开发最难的是职责的合理分配,或者叫:“如何合理的组织代码”,今天说一个关于这方面问题的示例,希望大家多批评。

示例背景

参考数据字典

需求

  1. OrderCode必须唯一。
  2. Total = Sum(Subtotal)。
  3. 订单有三种状态:【未提交】、【待审核】和【已审核】,合理的状态迁移有:【未提交】----》【待审核】和【待审核】----》【已审核】,只有处于【未提交】状态的订单能修改。
  4. 订单和订单项中的状态必须合法,规则自己定义。

示例实现

项目结构

  • Application:应用层,负责领域逻辑的封装。主要角色:ApplicationService、CommandHandler。
  • Boostrap:启动管理层,负责启动过程管理,如:注册Ioc、初始化配置。主要角色:BootstrapListener。
  • Commands:命令层,是一个契约层。主要角色:Comamnd、DTO。
  • Controllers:控制器层,边界层。主要角色:Controller。
  • Domain:领域层,负责领域逻辑的组织。主要角色:Aggregate、Entity、ValueObject、Factory、DomainService、IRepository、IUnitOfWork。
  • Events:事件层,是一个契约层,跨聚合流程可以采用。主要角色:Event。
  • EventSubscribers:事件监听层。主要角色:EventSubscriber。
  • Infrastructure:基础设施层。主要角色:Repository、QueryService、UnitOfWork。
  • Query:查询层,为UI的查询提供服务,主要角色:QueryService。

项目整体采用简单的CQRS架构,Command端采用DDD组织,Query直接从数据库返回dynamic类型。Event可以用来处理跨聚合通信,也可以用来处理长事务或离线事务。

重点介绍的领域层

采用状态模式处理状态迁移。

 using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Happy.Example.Domain.TestOrders
{
public interface ITestOrderState
{
void AddTestOrderDetail(TestOrderDetail testOrderDetail); void UpdateTestOrderDetail(Guid testOrderDetailId, decimal subtotal); void RemoveTestOrderDetail(Guid testOrderDetailId); void Commit(); void Verify(); string Status { get; } TestOrder TestOrder { set; }
}
}
 using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Happy.Example.Domain.TestOrders
{
public partial class TestOrder
{
private abstract class TestOrderState : ITestOrderState
{
public virtual void AddTestOrderDetail(TestOrderDetail testOrderDetail)
{
this.ThrowInvalidOperationException();
} public virtual void UpdateTestOrderDetail(Guid testOrderDetailId, decimal subtotal)
{
this.ThrowInvalidOperationException();
} public virtual void RemoveTestOrderDetail(Guid testOrderDetailId)
{
this.ThrowInvalidOperationException();
} public virtual void Commit()
{
this.ThrowInvalidOperationException();
} public virtual void Verify()
{
this.ThrowInvalidOperationException();
} public abstract string Status { get; } public TestOrder TestOrder { protected get; set; } private void ThrowInvalidOperationException()
{
throw new InvalidOperationException(string.Format("处于【{0}】的订单不能执行此操作!", this.Status));
}
}
}
}
 using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Happy.Example.Domain.TestOrders
{
public partial class TestOrder
{
private sealed class UnCommitted : TestOrderState
{
internal static readonly string UnCommittedStatus = "未提交"; public override void AddTestOrderDetail(TestOrderDetail testOrderDetail)
{
this.TestOrder.DoAddTestOrderDetail(testOrderDetail);
} public override void UpdateTestOrderDetail(Guid testOrderDetailId, decimal subtotal)
{
this.TestOrder.DoUpdateTestOrderDetail(testOrderDetailId, subtotal);
} public override void RemoveTestOrderDetail(Guid testOrderDetailId)
{
this.TestOrder.DoRemoveTestOrderDetail(testOrderDetailId);
} public override void Commit()
{
this.TestOrder.DoCommit();
} public override string Status
{
get { return UnCommittedStatus; }
}
}
}
}

采用封装集合手法处理Total的同步问题。

 using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; using Happy.Domain;
using Happy.Domain.Tree;
using Happy.Infrastructure.ExtentionMethods;
using Happy.Example.Events.TestOrders; namespace Happy.Example.Domain.TestOrders
{
public partial class TestOrder : AggregateRoot<Guid>
{
private ITestOrderState _orderState; protected TestOrder() { } public TestOrder(string orderCode, string customer)
{
orderCode.MustNotNullAndNotWhiteSpace("orderCode");
customer.MustNotNullAndNotWhiteSpace("customer"); this.Id = Guid.NewGuid();
this.OrderCode = orderCode;
this.Customer = customer;
this.OrderState = TestOrderStateFactory.CreateUnCommittedTestOrderState(this);
this.TestOrderDetails = new List<TestOrderDetail>();
} public virtual System.String OrderCode { get; protected set; }
public virtual System.String Customer { get; protected set; }
public virtual System.Decimal Total { get; protected set; }
public virtual System.String Status { get; protected set; }
public virtual ICollection<TestOrderDetail> TestOrderDetails { get; protected set; } private ITestOrderState OrderState
{
get
{
if (_orderState == null)
{
_orderState = TestOrderStateFactory.Create(this, this.Status);
} return _orderState;
}
set
{
_orderState = value;
this.Status = value.Status;
}
} public void AddTestOrderDetail(TestOrderDetail testOrderDetail)
{
this.OrderState.AddTestOrderDetail(testOrderDetail);
} public void UpdateTestOrderDetail(Guid testOrderDetailId, decimal subtotal)
{
this.OrderState.UpdateTestOrderDetail(testOrderDetailId, subtotal);
} public void RemoveTestOrderDetail(Guid testOrderDetailId)
{
this.OrderState.RemoveTestOrderDetail(testOrderDetailId);
} public void Commit()
{
this.OrderState.Commit();
} public void Verify()
{
this.OrderState.Verify();
} private void DoAddTestOrderDetail(TestOrderDetail testOrderDetail)
{
this.TestOrderDetails.Add(testOrderDetail); this.Total += testOrderDetail.Subtotal;
} private void DoUpdateTestOrderDetail(Guid testOrderDetailId, decimal subtotal)
{
var testOrderDetail = this.TestOrderDetails.First(x => x.Id == testOrderDetailId); this.Total -= testOrderDetail.Subtotal;
testOrderDetail.Subtotal = subtotal;
this.Total += testOrderDetail.Subtotal;
} private void DoRemoveTestOrderDetail(Guid testOrderDetailId)
{
var testOrderDetail = this.TestOrderDetails.First(x => x.Id == testOrderDetailId); this.TestOrderDetails.Remove(testOrderDetail);
this.Total -= testOrderDetail.Subtotal;
} private void DoCommit()
{
this.OrderState = TestOrderStateFactory.CreatePendingVerificationTestOrderState(this);
} private void DoVerify()
{
this.OrderState = TestOrderStateFactory.CreateVerifiedTestOrderState(this); this.PublishEvent(new TestOrderVerified());
}
}
}

效果图

背景

写这个简单的Demo过程,遇到了很多小的决策,这篇文章就看做一个开头吧,后边重点介绍每个决策点,在一篇文章中真的很难说完,喜欢看代码的朋友,先去https://happy.codeplex.com/下载。

DDD:订单管理 之 如何组织代码的更多相关文章

  1. RDIFramework.NET V3.3 WinForm版新增订单管理主从表事例

    功能描述 无论什么系统,除了常规的单表处理外,主从表的应用都是非常普遍的,RDIFramework.NET V3.3 WinForm版本中新增了一个主从表的事例供大家参考.主从表的界面设计大同小异,主 ...

  2. Flask实战-留言板-安装虚拟环境、使用包组织代码

    Flask实战 留言板 创建项目目录messageboard,从GreyLi的代码中把Pipfile和Pipfile.lock文件拷贝过来,这两个文件中定义了虚拟环境中需要安装的包的信息和位置,进入m ...

  3. 一个ssm综合小案例-商品订单管理----写在前面

    学习了这么久,一直都是零零散散的,没有把知识串联起来综合运用一番 比如拦截器,全局异常处理,json 交互,RESTful 等,这些常见技术必须要掌握 接下来呢,我就打算通过这么一个综合案例把这段时间 ...

  4. Unity3D如何有效地组织代码?(转)

    问题: Unity3D可以说是高度的Component-Based Architecture,同时它的库提供了大量的全局变量.如何来组织代码呢? 答: - Unity有一些自身的约定,譬如项目里的Ed ...

  5. 【共享单车】—— React后台管理系统开发手记:城市管理和订单管理

    前言:以下内容基于React全家桶+AntD实战课程的学习实践过程记录.最终成果github地址:https://github.com/66Web/react-antd-manager,欢迎star. ...

  6. Unity3D如何有效地组织代码?

    本文整理自知乎,原文链接:http://www.zhihu.com/question/21070379 问题: Unity3D可以说是高度的Component-Based Architecture,同 ...

  7. 使用 Git 来管理 Xcode 中的代码片段

    使用 Git 来管理 Xcode 中的代码片段 代码片段介绍 xcode4 引入了一个新 feature: code snippets,在整个界面的右下角,可以通过快捷键:cmd + ctrl + o ...

  8. 如何管理你的 Javascript 代码

    今天不聊技术的问题,咱们来聊聊在前端开发中如何管理好自己的 Javascript 代码.首先,咱们先来说说一般都有哪些管理方式?我相信  seajs . requirejs  对于前端开发者而言都不陌 ...

  9. Python基础-修改excel、redis、接口开发、组织代码

    pymysql模块补充内容 1. 游标.description():显示表的字段属性 (什么是游标:游标用于交互式应用,就好比word里的光标一样,要修改某个地方,要先把光标移动到这里) 用好这个方法 ...

随机推荐

  1. log4qt的使用

    Log4Qt替换成新版本使其支持Qt5:https://github.com/devbean/log4qt/tree/master/src/log4qt 1. 解压log4qt到目标文件夹,如D:\Q ...

  2. [推荐]WebService开发知识介绍

    [推荐]WebService开发知识介绍 WebService开发手册  http://wenku.baidu.com/view/df3992ce050876323112128a.html WebSe ...

  3. 为CentOS 6 配置本地YUM源

    在网上找了很多为CentOS 6配置本地YUM源的方法,其中有很多是与网络相关的,我只想配个自己用的,结果就发现这个方法比较简单实用,就转过来了. 环境:CentOS 6.0 默认的yum是以网络来安 ...

  4. 解决安卓SDK更新连不通问题

    http://wenku.baidu.com/link?url=d7t81OFF4_o2YF9iBne-azyovROGPGOozMgWKNyAIQK8vtI0mIjvzpfdOXg7KOobu202 ...

  5. java中static{}语句块详解

    static{}(即static块),会在类被加载的时候执行且仅会被执行一次,一般用来初始化静态变量和调用静态方法,下面我们详细的讨论一下该语句块的特性及应用. 一.在程序的一次执行过程中,stati ...

  6. Mule ESB 社区版 企业版 资源下载 包含3.5和3.6

    很多的资源官方已经没有提供下载了,我将资源上传到网盘,供大家下载和收藏 AnypointStudio-for-win-32bit-5.0.2-201502251307.ziphttp://pan.ba ...

  7. React同构直出原理浅析

    通常,当客户端请求一个包含React组件页面的时候,服务端首先响应输出这个页面,客户端和服务端有了第一次交互.然后,如果加载组件的过程需要向服务端发出Ajax请求等,客户端和服务端又进行了一次交互,这 ...

  8. json字符串转成 Map/List

    package jsonToMap; import java.util.List; import java.util.Map; import java.util.Map.Entry; import n ...

  9. jquery实现返回基部案例效果

    <!doctype html> <html> <head> <meta charset="gb2312"> <title> ...

  10. SVO实时全局光照优化(里程碑MK0):Sparse Voxel Octree based Global Illumination (SVO GI)

    完全自主实现,bloat-free.再次声明,这不是UE.U3D.CE.KlayGE! 老规矩,先贴图.后面有时间再补充描述. 1. 支持多跳间接全局光照2. 支持vxao/so.vxdiff/spe ...