Account的简单架构
前几天,有园友私下问我,博客中的AccountDemo后端架构为什么是那样的,是不是分层太多太冗余,故这里简单介绍下。先看解决方案工程截图:
每个工程的含义,见https://www.cnblogs.com/guokun/p/7082987.html。运行时,请求处理流程,这里再贴出来:
这里,园友主要不解的是,为什么会有两个接口层,Account.Repository.Contract和Account.Service.Contract,是不是太多了。最近几年,在后端架构中,出现了一种称之为六边形架构的架构模式,这货之前曾被叫做洋葱架构、端口适配器架构,反正大家知道都是它就是了。六边形架构的核心,就是应用程序业务逻辑处于架构的核心,而上层的视图、控制器、数据访问等,都属于基础设施,是用来辅助实现业务逻辑的,他们都依赖于核心业务逻辑。这些基础设施是易变或者说很可能被频繁替换的,例如应用层今天可能是MVC,明天可能是WebAPI,数据访问今天可能是EF,明天可能是Dapper,甚至CSRedis,MongoDB。。。
六边形架构最终要实现的效果就是,解耦应用核心业务逻辑与基础设施,其整体架构与依赖如下图:
蓝色箭头方向代表依赖方向,而非运行时数据流向或请求处理流向,请特别注意。ApplicationCore处于整个架构的中心,周边都是依赖于它的,这也是这一层名称ApplicationCore的由来,其核心特征是:1、用用层及基础设施层都依赖核心业务层;2、业务逻辑保持不变,应用层或基础设施层,比如切库、切ORM、切应用层框架,随便搞;3、有别于传统三层架构,数据层提供什么,业务层就有什么或用什么,六边形架构是业务层需要什么,就定义什么契约,数据层就实现什么或提供什么。
介绍完了六边形架构,接下来回答,为什么有两个接口层。本质上,Account.Repository.Contract和Account.Service.Contract两层契约均归属于核心业务层,Account.Service.Contract用于对应用层承诺,提供什么服务,Account.Repository.Contract和Account.Service.Contract则规定基础设施层必须给自己提供什么操作。如果你愿意,那么这两个接口层完全可以融入Account.Service工程中,这都是没问题的,本来他们就属于业务逻辑的范畴,但我还是把它单独分出来,否则便是应用层、基础设施层直接硬依赖Account.Service,一者太重,二者不符合将面向接口编程。
最后,说下,为什么Account.Repository.EF仓储工程中,一个实体类,对应了一个仓储对象。严格来讲,这么做是不合适的,设想一下,假如数据库表很多,那这里岂不膨胀得厉害。要弄明白这个问题,首先得知道仓储的由来。这玩意儿来自领域驱动架构,一般来讲,一个仓储是一一对应一个聚合根,这个聚合根是业务上功能聚合的一系列领域对象的,例如一个学生,对应一个宿舍,同时这个学生是个高富帅,他他妈的比较花心,身边有N个白富美女朋友。如果系统要维护这么样一个对应关系或信息,这里学生就是一个聚合根。具体表现在代码中,直观看就类似一个复杂对象,这个复杂对象的最外边就是学生,里边嵌套啥宿舍啊,女朋友集合啊,什么的。正常情况下,应该是学生这个聚合根才对应一个仓储类的,什么宿舍,女朋友都不应该有仓储类(假设没有其他需求导致他们需要上升为聚合根)。解释完了聚合根,这里回到刚才那问题,为什么搞成了一个数据库实体一个仓储类。主要在于,示例中抽象出了这么一个仓储基类:
这玩意儿是泛型的,因为后续仓储实现类想要用到其中的一些公用方法,实现这个基类时候,需要约定实体,所以为了偷懒,我就每个数据库表或者领域实体一个仓储类了,仅此而已。
好了,园友提到的几个问题差不多就这样。
Account的简单架构的更多相关文章
- android APP是否需要缓存?+简单架构
问题的由来 昨天,当我写完我的第一篇博客之后,我便百无聊赖的玩起了手机!当我打开Google自带的一些app的时候,发现他们直接叫我连接网络,并没有缓存上次从网络获取的数据.这就让我感到很奇怪!于是我 ...
- PHP网站简单架构 – 单独跑php-fpm
这个架构比较简单,不做过多的说明 前端1台Nginx:负载均衡+nfs 中间2台php:php-fpm 后端1台数据库:MySQL 安装略,参考<lnmp最新源码一键安装包> 192.16 ...
- 简单架构:反射实现抽象工厂+IDAL接口完全独立DAL
一.普通架构中存在的问题 StudentDB数据库,包含一张StudentInfoTB表,结构如下: s_id int primary key identity(1,1), s_name Nvarch ...
- 重构 JAVA 聊天室 —— CS 模式的简单架构实现
前言 自从开始弄起数据挖掘之后,已经很久没写过技术类的博客了,最近学校 JAVA 课设要求实现一个聊天室,想想去年自己已经写了一个了,但是有些要求到的功能我也没实现,但看着原有的代码想了想加功能好像有 ...
- [转]分享php中四种webservice实现的简单架构方法及实例
FROM : http://www.itokit.com/2012/0417/73615_2.html 本人所了解的webservice有以下几种:PHP本身的SOAP,开源的NUSOAP,商业版的P ...
- 分享php中四种webservice实现的简单架构方法及实例
一:PHP本身的SOAP所有的webservice都包括服务端(server)和客户端(client).要使用php本身的soap首先要把该拓展安装好并且启用.下面看具体的code首先这是服务端实现: ...
- 分享php中四种webservice实现的简单架构方法及实例(转)
本人所了解的webservice有以下几种:PHP本身的SOAP,开源的NUSOAP,商业版的PHPRPC,以及使用二进制传输数据流的 HessianPHP,那么一下就简单的介绍下这几种webserv ...
- 分享php中四种webservice实现的简单架构方法及实例[转载]
[转载]http://www.itokit.com/2012/0417/73615.html 本人所了解的webservice有以下几种:PHP本身的SOAP,开源的NUSOAP,商业版的PHPRPC ...
- web开发之负载均衡的简单架构
负载均衡 负载均衡的核心思想就是:请求分担 最简单的配置: 一台负载均衡服务器 两台webserver服务器 两台webserver服务器需要配置相同的服务器环境,设置相同的域名指向 负载均衡服务器需 ...
随机推荐
- 【洛谷P3224】永无乡 并查集+Splay启发式合并
题目大意:给定 N 个点的图,点有点权,初始有一些无向边,现在有 Q 个询问,每个询问支持动态增加一条无向边连接两个不连通的点和查询第 X 个点所在的联通块中权值第 K 大的是哪个点. 题解:学会了平 ...
- spring 项目分开发和生产环境
1.pom 文件修改 <profile> <!-- 本地开发环境 --> <id>dev</id> <properties> <pro ...
- DoubleOps.java
/****************************************************************************** * Compilation: javac ...
- java.lang.NoClassDefFoundError: org/junit/runner/manipulation/Filter
今天想写个随笔,最近经常遇到使用junit的时候报java.lang.NoClassDefFoundError,今天算是恍然大悟了,原来junit虽然在gradle里面配置了,也在Project an ...
- Linux发行版Debian操作系统破译密码
Linux发行版Debian操作系统破译密码 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 其实玩过Linux的小伙伴,对破解服务器密码都应该有所了解,典型的两个代表,我觉得一个是 ...
- nginx错误集合
遇到 nginx: [warn] server name "http://127.0.0.1" has suspicious symbols in D:\nginx-1.12.1/ ...
- 注解 和 xml 配置的优缺点【转】
java annotation(注解) 的优点缺点 Annotation和xml各自作为配置项的优点与缺点. Annotation 一.Annotation 的优点 1.保存在 class 文件中,降 ...
- 把svn上的mycelipse导到本地的eclipse中【原】
myeclipse和eclipse的web项目互导时会产生各种问题,现在把我遇到的情况记录如下: eclipse如何把svn上down下来的myeclipseWeb项目变成eclipse的Web项目: ...
- C# 实现线段的编码裁剪算法(vs2010)
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...
- springboot系列之-logging
配置文件以application.yml为例说明: Spring Boot默认的日志组件为Logback. 一. 日志配置参数: logging: file: # 日志文件,绝对路径或相对路径 pat ...