1、DDD领域驱动设计实践篇之如何提取模型

2、DDD领域驱动设计之聚合、实体、值对象

3、DDD领域驱动设计之领域基础设施层

4、DDD领域驱动设计之领域服务

5、整体DEMO代码

什么是运用层,说白了就是以前三层的BLL,没有什么特别,只是调用的不是以前的DAL了,而是领域层+基础设施层,运用层接口基本是可以根据原型来做的,也就是UI需要什么数据就定义什么接口,难点就在于怎么去调用领域层了,这个如果是分开来开发的话,难度会很大,为什么是很难呢,原因就在于区分职责,需要有很详细的领域层说明以及实时的参与领域建模。

运用层代码1

using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using DDD.Domain;
using DDD.Domain.Arrange; namespace DDD.Services
{
public class PlanArrangeService : NDJHService<PlanArrange>, IPlanArrangeService
{
private readonly IPlanArrangeRepository repository;
private readonly ArrangeService service; protected PlanArrangeService(IPlanArrangeRepository repository, ArrangeService service)
: base(repository)
{
this.repository = repository;
this.service = service;
} /// <summary>
/// 查找可下发的批次列表
/// </summary>
/// <param name="xzqdm"></param>
/// <returns></returns>
public override IList<BatchModel> FindAllowSendBatches(string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
if (!xzqdm.EndsWith("00")) //如果是县级用户,则不需要判断t.XFXZQDM != t.XZQDM这个条件
{
var list = FindBy(0, 0, xzqdm, ProjectStauts.Default).OrderBy(a => a.APPC).ToList();
var query = from t in list
group t by new { t.APPC, t.ND }
into temp
select new BatchModel
{
Batch = temp.Key.APPC,
IsSent = false,
Year = temp.Key.ND
};
return query.ToList(); }
else
{
return base.FindAllowSendBatches(xzqdm);
} } /// <summary>
/// 查找计划安排列表
/// </summary>
/// <param name="year">年度</param>
/// <param name="grade">指标级别</param>
/// <param name="xzqdm">行政区代码</param>
/// <returns></returns>
public IList<PlanArrange> FindByDescendants(int year, string xzqdm, IndicatorGrade? grade = null)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
var query = this.repository
.Where(GetDistrictCodeExpr(xzqdm))
.Where(grade.HasValue, a => a.ZBJB == grade.Value)
.Where(year != 0, a => a.ND == year);
if (xzqdm.EndsWith("0000"))
{
//省级用户可以查看国家安排、省级安排、市县级已下发的安排
if (grade == IndicatorGrade.City || grade == IndicatorGrade.County)
{
query = query.Where(t=>t.Status == ProjectStauts.Sent);
}
}
else if (xzqdm.EndsWith("00"))
{
//市级用户可以查看国家、省、县已下发的安排、市级安排
if (grade != IndicatorGrade.City)
{
query = query.Where(t => t.Status == ProjectStauts.Sent);
}
}
else
{
//县级用户可以查看国家、省、市已下发的安排、县级项目
if (grade != IndicatorGrade.County)
{
query = query.Where(t => t.Status == ProjectStauts.Sent);
}
}
return query.OrderBy(t => t.APPC).ToList();
} /// <summary>
/// 行政区代码表达式
/// </summary>
/// <param name="xzqdm">一般为参数传递的值</param>
/// <returns></returns>
private Expression<Func<PlanArrange, bool>> GetDistrictCodeExpr(string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
if (xzqdm.EndsWith("0000"))
{
var sub = xzqdm.Substring(0, 2);
return d => d.XFXZQDM.Contains(sub);
}
else if (xzqdm.EndsWith("00"))
{
var sub = xzqdm.Substring(0, 4);
return d => d.XFXZQDM.Contains(sub);
}
else
{
return d => d.XFXZQDM.Contains(xzqdm);
}
} public override void Update(PlanArrange entity)
{
service.CheckUpdate(entity);
base.Update(entity);
} public override void Register(PlanArrange entity, IList<PlanArrange> entities)
{
if (entity == null) throw new ArgumentNullException("entity");
Validate(entity);
Validate(entities);
service.Register(entity, entities);
repository.Add(entity);
foreach (var item in entities)
{
repository.Add(item);
}
}
}
}

运用层代码2

using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Text;
using System.Threading.Tasks;
using DDD.Domain;
using DDD.Domain.Indicator; namespace DDD.Services
{
public class PlanIndicatorService : NDJHService<PlanIndicator>, IPlanIndicatorService
{
private readonly IPlanIndicatorRepository repository;
private readonly IndicatorService service; protected PlanIndicatorService(IPlanIndicatorRepository repository, IndicatorService service)
: base(repository)
{
this.repository = repository;
this.service = service;
} /// <summary>
/// 查找指标列表(国家下发,省级发,市级下发)
/// </summary>
/// <param name="year">年度</param>
/// <param name="xzqdm">下发行政区代码</param>
/// <param name="grade">指标级别</param>
/// <returns></returns>
public IList<PlanIndicator> FindReceive(int year, string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return this.repository
.Where(t => t.XFXZQDM != t.XZQDM)
.Where(t => t.XFXZQDM == xzqdm)
.Where(t => t.Status == ProjectStauts.Sent)
.Where(year != 0, a => a.ND == year)
.OrderBy(a => a.APPC).ToList();
} /// <summary>
/// 查找预留指标列表(省级预留,市级预留)
/// </summary>
/// <param name="year">年度</param>
/// <param name="xzqdm">行政区代码</param>
/// <returns></returns>
public IList<PlanIndicator> FindReserve(int year, string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return this.repository
.Where(a => a.XFXZQDM == a.XZQDM)
.Where(a => a.XZQDM == xzqdm)
.Where(year != 0, a => a.ND == year)
.OrderBy(a => a.APPC).ToList();
} /// <summary>
/// 查找下发指标列表(下发市级,下发县级)
/// </summary>
/// <param name="year">年度</param>
/// <param name="xzqdm">行政区代码</param>
/// <returns></returns>
public IList<PlanIndicator> FindSendTo(int year, string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return this.repository
.Where(a => a.XFXZQDM != a.XZQDM)
.Where(a => a.XZQDM == xzqdm)
.Where(year != 0, a => a.ND == year)
.OrderBy(a => a.APPC).ToList();
} /// <summary>
/// 查找指定行政区下发的指标(预留+下发)
/// </summary>
/// <param name="status">状态</param>
/// <param name="xzqdm">行政区代码</param>
/// <returns></returns>
public IList<PlanIndicator> FindSend(string xzqdm, ProjectStauts status)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return this.repository
.Where(a => a.XZQDM == xzqdm)
.Where(t => t.Status == status)
.OrderBy(a => a.APPC).ToList();
} /// <summary>
/// 统计预留指标
/// </summary>
/// <param name="year"></param>
/// <param name="xzqdm"></param>
/// <returns></returns>
public IndicatorArea TotalReserveIndicator(int year, string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return repository.TotalReserveIndicator(year, xzqdm);
} /// <summary>
/// 统计上级下发给下级指标(省-国家下发,市-省级下发,县-市级下发)
/// </summary>
/// <param name="year"></param>
/// <param name="xzqdm"></param>
/// <returns></returns>
public IndicatorArea TotalReceiveIndicator(int year, string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return repository.TotalReceiveIndicator(year, xzqdm);
} /// <summary>
/// 统计下发给下级的指标(省-市级下发,市-县级下发)
/// </summary>
/// <param name="year"></param>
/// <param name="xzqdm"></param>
/// <returns></returns>
public IndicatorArea TotalSendToIndicator(int year, string xzqdm)
{
if (string.IsNullOrEmpty(xzqdm)) throw new ArgumentNullException("xzqdm");
return repository.TotalSendToIndicator(year, xzqdm);
} public override void Update(PlanIndicator entity)
{
service.CheckUpdate(entity);
base.Update(entity);
} public override void Register(PlanIndicator entity, IList<PlanIndicator> entities)
{
if (entity == null) throw new ArgumentNullException("entity");
Validate(entity);
Validate(entities);
service.Register(entity, entities);
repository.Add(entity);
foreach (var item in entities)
{
repository.Add(item);
}
}
}
}

DDD领域驱动设计之运用层代码的更多相关文章

  1. C#进阶系列——DDD领域驱动设计初探(七):Web层的搭建

    前言:好久没更新博客了,每天被该死的业务缠身,今天正好一个模块完成了,继续来完善我们的代码.之前的六篇完成了领域层.应用层.以及基础结构层的部分代码,这篇打算搭建下UI层的代码. DDD领域驱动设计初 ...

  2. DDD领域驱动设计之领域基础设施层

    1.DDD领域驱动设计实践篇之如何提取模型 2.DDD领域驱动设计之聚合.实体.值对象 其实这里说的基础设施层只是领域层的一些接口和基类而已,没有其他的如日子工具等代码,仅仅是为了说明领域层的一些基础 ...

  3. DDD领域驱动设计初探(七):Web层的搭建

    前言:好久没更新博客了,每天被该死的业务缠身,今天正好一个模块完成了,继续来完善我们的代码.之前的六篇完成了领域层.应用层.以及基础结构层的部分代码,这篇打算搭建下UI层的代码. DDD领域驱动设计初 ...

  4. DDD 领域驱动设计-谈谈 Repository、IUnitOfWork 和 IDbContext 的实践(2)

    上一篇:<DDD 领域驱动设计-谈谈 Repository.IUnitOfWork 和 IDbContext 的实践(1)> 阅读目录: 抽离 IRepository 并改造 Reposi ...

  5. DDD 领域驱动设计-如何控制业务流程?

    上一篇:<DDD 领域驱动设计-如何完善 Domain Model(领域模型)?> 开源地址:https://github.com/yuezhongxin/CNBlogs.Apply.Sa ...

  6. DDD 领域驱动设计-如何完善 Domain Model(领域模型)?

    上一篇:<DDD 领域驱动设计-如何 DDD?> 开源地址:https://github.com/yuezhongxin/CNBlogs.Apply.Sample(代码已更新) 阅读目录: ...

  7. DDD领域驱动设计之领域服务

    1.DDD领域驱动设计实践篇之如何提取模型 2.DDD领域驱动设计之聚合.实体.值对象 3.DDD领域驱动设计之领域基础设施层 什么是领域服务,DDD书中是说,有些类或者方法,放实体A也不好,放实体B ...

  8. DDD 领域驱动设计-谈谈 Repository、IUnitOfWork 和 IDbContext 的实践(1)

    好久没写 DDD 领域驱动设计相关的文章了,嘎嘎!!! 这几天在开发一个新的项目,虽然不是基于领域驱动设计的,但我想把 DDD 架构设计的一些东西运用在上面,但发现了很多问题,这些在之前的短消息项目中 ...

  9. C#进阶系列——DDD领域驱动设计初探(一):聚合

    前言:又有差不多半个月没写点什么了,感觉这样很对不起自己似的.今天看到一篇博文里面写道:越是忙人越有时间写博客.呵呵,似乎有点道理,博主为了证明自己也是忙人,这不就来学习下DDD这么一个听上去高大上的 ...

随机推荐

  1. {POJ}{动态规划}{题目列表}

    动态规划与贪心相关: {HDU}{4739}{Zhuge Liang's Mines}{压缩DP} 题意:给定20个点坐标,求最多有多少个不相交(点也不相交)的正方形 思路:背包问题,求出所有的正方形 ...

  2. jquery ajax load

    jQuery load() 方法 jQuery load() 方法是简单但强大的 AJAX 方法. load() 方法从服务器加载数据,并把返回的数据放入被选元素中. 语法: $(selector). ...

  3. nginx rewirte

    server { listen 8888; server_name jobPhp; root F:\ck\Porject\quanRelease\quanJob; try_files $uri $ur ...

  4. UBUNTU9.10下安装TFTP学习笔记一(arm学习SEED-138板子)

    擦,刚刚写的没保存都丢了,郁闷中~~~~ 简单重写 1什么是TFTP .安装TFTP(TFTP(Trivial File Transfer Protocol,简单文件传输协议)是TCP/IP协议族中的 ...

  5. web.py+mysql插入中文提示query = query.encode(charset) UnicodeEncodeError: 'latin-1' codec can't encode characters in position 86-100

    对于中文编码的问题,总会出现各种各样恶心的错误,还不知道应该怎么解决,首先,你从最开头就应该关注编码问题,尽量保证所有的编码方式都是一致的 用python+web.py+mysql来写程序,首先要保证 ...

  6. oracle 同时更新(update)多个字段多个值

    --创建表A,B: create table A (a1 varchar2(33),a2 varchar2(33),a3 varchar2(33)); create table B (b1 varch ...

  7. SQL数据库完全复制

    很少摸 Windows 环境下的东西,最近被个 MS SQL Server 的数据库搞得头大.实在不像 MySQL 那样用起来轻车熟路, OrZ ... 本来以为企业管理器里面既然提供了 DTS 数据 ...

  8. java多线程学习-同步之线程通信

    这个示例是网上烂大街的,子线程循环100次,主线程循环50次,但是我试了很多次,而且从网上找了很多示例,其实多运行几次,看输出结果并不正确.不知道是我转牛角尖了,还是怎么了.也没有大神问,好痛苦.现在 ...

  9. UIImageView、UISlider、UISwitch、UIStepper、UISegmentControl

    UIImageView——图像视图 作用:专门用来显示图片的控件 . 设置图像 [self.imageView setImage:[UIImage imageNamed:@"abc.png& ...

  10. 连接Oracle的帮助类

    package util; import java.sql.Connection;import java.sql.DriverManager;import java.sql.PreparedState ...