今儿写这个题目胆子有点大,不过还是得冒险整一篇(我怕您看完了骂我),一是出于经验分享,另外则是为了后面我们讲案例的时候做好铺垫。好的代码需要注意的事项其实挺多的,您真让我一骨脑儿都列出来可能也差点意思,所以遵照我们常态化歪楼的习惯,我是想到哪写到哪儿。
我没事儿的时候就喜欢看别人写的文章,也喜欢看书,收获还是挺多的,不过岁数大了忘性也大,记不住。可有一个事情我记得倍儿清楚:咱们搞技术的尤其是后端开发都知道一类对象叫视图模型(View Object简称VO),一说VO,大多数文章给的概念就是“后端传向前端,用于承载需要在页面上显示的信息的实体或对象(反方向是前端传向后端用于存储的信息)”。概念虽然简单,可是和我的想法就不一样,谁叫咱这人个色呢,您听我给您唠唠。
我所认为的视图相对来说要广义一点:前面您所认为的自然是没问题;还有一种(咱们以Java为例啊)我来举例说明:订单的服务需要调用客户服务来查询客户信息“CustomInfo”,这里面的“CustomInfo”就是视图模型。您肯定暴起反对:搞笑呢吧?“CustomInfo”是“DTO(数据传输对象)”。这种说法当然没错了,但DTO这个说法太广义面且太模糊了。常用的数据实体也算是一种DTO,他承载了需要存储和从数据库查询返回的数据;前端传向后端的也是DTO,承载了用户的输入。所以您平常老是用DTO这个说法其实真的不够准确。那为什么说服务间相互传的数据是VO呢?比如您去相亲,对方给您的印象(注意:印象不仅是你主动获取的,还包括对方传给你的,就跟两个函数一样你调用我我调用你)比如长相、谈吐等这些是对方想让您知道的(别反对,女人画起妆来你就不知道她到底长什么样)关于其自身属性的部分信息,当然还有一部分是您从对方身上获得的信息,所谓的“印象”是两种信息的集成,其实就是信息视图。所以您调用下游服务时对方返回给你的就是这个下游服务的视图也就是下游服务想要展示给你的内容。而且,不仅仅是服务间有视图,包与包之前也只能通过视图了解彼此。
上一段我提到的“服务间和包间只能通过视图了解彼此”,引申的含义是说:服务间和包间只能通过视图传递信息,这种限制不论是3层还是4层构架都适用。DTO这种称呼相对模糊,一般我在开发的时候也不会这么叫。实际上,您是喜欢叫VO还是DTO也不耽误什么事儿,可我们在这里给出了一个重要的约束也就是包或系统间的数据访问限制。这东西特别容易出乱子,尤其是想把一个服务做二次拆分的时候,如果当前系统无访问约束那拆的风险就会相当的高。
除了视图模型外,后端服务开发还会用到另外两类:数据模型和领域模型,如下图所示。您别看统共就三种,想用好了没那么容易。至少在我经历的不少项目中很多人都是乱用的,要不是没有访问限制要不就是模型冗余。

|
重点! 服务间和包间相互调用时,传入和传出的信息(简单字段除外)只能通过视图模型进行承载,不得将数据模型和领域模型作为传输信息的载体,包括在消息中也不可以内嵌领域与数据模型,以避免内部信息泄露。 |
对于模型的滥用,让我来厚着脸皮做一下纠偏,分别说一个每个对象的作用和主要的使用限制。看完了后您会觉得:“这也太夸张了,写个代码有那么麻烦吗?”,答:有!好东西一般不会是多快好省出来的。
定义:描述数据库设计并承载数据在持久层到应用层间之间的传输。限制:1)每一张表一个实体;2)级联查询的结果一般是多个表的集成,也需要建立对应的实体;3)不可以传到包、服务之外;4)不要包含任何业务逻辑;5)DAO的输入输出只能是简单字段或数据模型;6)仅能使用基本类型如Integer、String等。有一个小技巧:如果数据模型和视图模型的字段一样,赋值时可以使用一些工具如“BeanUtils”实现两个对象间的字段复制。
定义:用于系统间、模块间、包间传送和展示数据的载体,具体的解释可参考上面内容。限制:1)仅包含最少的用于传输的信息,不要使用一个对象包含所有的字段即万能对象;2)不可以从数据模型继承(这个问题尤其普遍),可使用工具实现与数据模型的互转;3)是包、服务间传输信息的唯一载体;4)不得包含业务逻辑。下面给出了一个“数据字典”视图模型的示例,请参考。
@ApiModel(value = "数据字典信息")
public class DictionaryVO extends VOBase {
private static final int MAX_VALUE_LENGTH = 32;
@ApiModelProperty(value = "字典值,长度:32", required = true)
private String value;
@Override
public ParameterValidationResult validate() {
if (this.classId == null) {
return ParameterValidationResult.failed(OperationMessages.INVALID_CLASS_ID);
}
if (StringUtils.isEmpty(this.value) || this.value.length() > MAX_VALUE_LENGTH) {
return ParameterValidationResult.failed(String.format(OperationMessages.INVALID_VALUE_LENGTH, MAX_VALUE_LENGTH));
}
return super.validate();
}
public static DictionaryVO of(DictionaryDataEntity entity) {
if (entity == null) {
return null;
}
DictionaryVO vo = new DictionaryVO();
BeanUtils.copyProperties(entity, vo);
return vo;
}
}
上述代码中,视图模型继承于“VOBase”自定义类,此类包含了“validate”方法用于对视图模型中的信息进行验证。切记:前端、其它服务和包传过来的信息永远是不可信的,通过在视图模型中增加验证逻辑可以让代码更简洁。“of”方法用于数据模型和视图模型的转换。“ApiModel”引入了“Swagger”用于对字段进行说明。
定义:描述业务实体属性和行为的模型。一般来说是充血模型,后续会细讲。限制:1)不得依赖于架构中的其它层、第三方基础框架如Spring、DAO、HTTPClinet等。这么说吧,除了JDK外其它都不能依赖。就和狗一样,依赖少表示血统纯粹,越纯粹越好;2)作用范围只能在业务模型层中,不得外泄;3)严格注意每个模型的访问限制,能不用public就不用。4)最好自行写一些包含了公共能力(比如“对象判等”)的领域模型基类来保证代码的干净。
这个里面我觉得有可能争议最大的是关于领域模型的依赖,有一些比如字符串工具、日期工具这种的第三方类库其实很普遍,完全的不依赖是否会更加的极端?怎么说呢,我个人在使用ODD开发的时候写了一套基础组件,包含了验证、值类型、实体类型、领域仓库、Saga等领域模型相关的组件(大部分都是抽象类)和少量的工具类,需要什么拿来即用,我称之为“通用能力库”。这个库本身是自己写的(注:本系列文章只关注DDD知识的讲解,不会推荐任何的、成熟度不够高的尤其是标榜为DDD的框架),也的确没用到JDK外的其它第三方组件。而且,领域模型本身专用于业务逻辑处理和计算,像什么字符串格式化啊、日期格式化啊其实就不应该在领域模型里搞。写通用能力库毕竟也会占用精力,我的推荐是您在领域模型中尽量少的依赖第三方组件,越少越好。有些书籍中会展示在领域模型中注入DAO来实现嵌套对象的“懒加载”,我个人认为这个是不化不类,只有在需要向性能妥协时才会使用。
本章讲了三个模型,虽然内容不多,可真正使用起来也是有一定的要求的。软件开发是个细活儿,想做好当然要负出精力了。您完全可以突破上述所提到的限制,也能出东西,可需求总是变化的,总得改代码,到时候吃亏的还是自己。这没办法,坑儿是您自己给自己挖的,除非您写完了代码就打算跑路,那也给了后面接手的人骂爹的机会。另外,您可能认为本章和DDD没关系,我得提前声明:咱不是写小说呢,一个字多少多少钱,这些经验您还真在其它书上找不着。而且,越是细节越能体现开发者的能力,搞技术到后面比的是什么?不再是会什么不会什么了,而是比谁开发的质量高和可维护性高。两个员工做同样的需求尤其是功能增强性相关的,一个5分钟搞定,一个3天,你是老板你用哪个 ?
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭10年前。问题1)我想知道rubyonrails是否有功能类似于primefaces的gem。我问的原因是如果您使用primefaces(http://www.primefaces.org/showcase-labs/ui/home.jsf),开发人员无需担心javascript或jquery的东西。据我所知,JSF是一个规范,基于规范的各种可用实现,prim
本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01 客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02 数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit
目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称
项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU
我在我的项目中有一个用户和一个管理员角色。我使用Devise创建了身份验证。在我的管理员角色中,我没有任何确认。在我的用户模型中,我有以下内容:devise:database_authenticatable,:confirmable,:recoverable,:rememberable,:trackable,:validatable,:timeoutable,:registerable#Setupaccessible(orprotected)attributesforyourmodelattr_accessible:email,:username,:prename,:surname,:
我正在尝试创建密码规则来设计可恢复的密码更改。我通过passwords_controller.rb做了一个父类(superclass),但我需要在应用规则之前检查用户角色,但我所拥有的只是reset_password_token。 最佳答案 假设您的模型是用户:User.with_reset_password_token(your_token_here)Source 关于ruby-on-rails-设计通过reset_password_token获取用户,我们在StackOverflow
我已经使用Apartment设置了一个Rails5应用程序(1.2.0)和Devise(4.2.0)。由于某些DDNS问题,应用只能在app.myapp.com下访问(请注意子域app)。myapp.com重定向到app.myapp.com。我的用例是每个注册该应用的用户(租户)都应该通过他们的子域(例如tenant.myapp.com)访问他们的特定数据。用户不应限定在其子域内。基本上应该可以从任何子域登录。重定向到租户的正确子域由ApplicationController处理。根据Devise标准,登录页面位于app.myapp.com/users/sign_in。这就是问题开始的
我在关注RyanbatesRailsCast的devise和omniauth(第235集-devise-and-omniauth-revised)。当我尝试使用Twitter登录时,标题中不断出现错误。defself.new_with_session(params,session)ifsession["devise.user_attributes"]new(session["devise.user_attributes"],without_protection:true)do|user|user.attributes=paramsuser.valid?end完整跟踪:C:/Ruby20