- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在重构我们的 BI 层以使我们的代码更加“松散耦合”,我很想听听你们认为可能会做出哪些有趣的改进?
目前,我们的 API 是这样的:-
// Fetch a collection of objects
ProductCollection prods = Product.GetProducts();
// Load an individual object, make change and save it back
Product p = new Product();
if(p.Load(productID))
{
p.Name = "New Name";
p.Save();
}
如您所见,我们用于获取对象集合/加载单个对象和保存更改的方法都内置在“模型”类中。我们的每个模型类都继承自 ObjectBase 基类,该基类包括 DB 访问函数和更改跟踪,因此当有人通过属性更改值时,对象会自动标记为脏,并向订阅了这些事件的任何对象 (UI) 触发通知。
我想做的是使用“存储库模式”,以便我们可以将数据库实现从模型中抽象出来。但是,我一直在查看的大部分代码似乎都表明“模型”类不应该包含任何智能,而应该只是数据的容器。相反,应该通过使用服务来应用逻辑。那么这是否意味着完成上述我需要做类似的事情
List<Product> prods = ProductService.GetProducts();
Product p = ProductService.GetSingleProduct(productID);
p.Name = "New Name";
ProductService.SaveProduct(p);
这似乎是一种更复杂的方法,并且更难将功能封装在业务对象中。
有人可以解释为什么这是一种更好的方法,或者我可能误解了这些概念吗?
谢谢
詹姆斯
最佳答案
您当前的 API 是 Active Record Pattern 的实现。 .当代码中使用的对象模型与数据库模型一对一匹配时,这种模式往往可以正常工作。另一个优点是存在生成这些类的工具,包括持久性代码和数据库表。
您建议的替代方案是存储库模式。正如您已经提到的那样,实现这有点复杂,但有几个优点。由于您可以实现任何类型的 ORM 工具,因此您不仅限于一对一的映射,还可以实现更复杂的映射,其中对象模型可能与数据库模型不同。因此,您不必在数据库中强制使用对象模型或相反。然而,除了一对一之外,更复杂的映射无法生成,需要一些
另一个优点是您可以更轻松地创建测试,因为您可以创建一个甚至不需要数据库的 Mock 存储库。
使用存储库模式,您还可以将模型与持久性逻辑分开。
在这两种情况下,都可以以通用方式编写持久性方法,以便持久性代码是通用的,不需要知道需要持久化的特定对象。这对于 Active Record 模式来说是显而易见的,因为所有这些对象都实现了保存、删除、更新等。对于存储库模式,您还可以使用适用于任何对象的 ORM 工具,这样就可以编写这样的代码:
Repository.Save(ObjectOfAnyType);
ObjectOfAnyType 可以是任何东西,只要 ORM 工具为对象的类型定义/实现了一些映射。
因此,您可以选择、想要或需要这些优势,但会增加一点复杂性。或者事件记录模式的简单性就足够了。
我总是倾向于使用存储库模式,但有时也会使用 Active Record 模式,主要用于快速原型(prototype)设计。
关于architecture - 改变我们的 BL 设计——哪种模式最有用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/386315/
当我开始学习一门新语言时,我总是觉得我没有以实用、标准的方式进行学习。所以这里有一个关于 jQuery 的问题以及我所做的是否可以接受。 我有 3 张图片。 然后我让 jQuery 检测 $('
基本上,我想知道线程是否有用或必要,或者可能更具体地说,您将使用它的用途和情况。我对线程了解不多,也从未使用过它(我主要使用 C#),并且想知道如果使用它们是否会提高性能或稳定性。如果有人愿意解释一下
这个问题在这里已经有了答案: What is The Rule of Three? (8 个答案) 关闭 7 年前。 嘿嘿。我有一个让我很难过的问题。我自定义了一个普通的拷贝构造函数但它只在我初始化
这个问题在这里已经有了答案: 关闭 12 年前。 Possible Duplicate: Uses for multiple levels of pointer dereferences? 我在 C
我不确定异常在每种语言中的工作方式是否相同,但我使用的是 PHP,我想知道我什么时候做这样的事情: if (!$this->connection[0]->query($this->query)) t
Scala native 是最近发布的,但是他们(现在)使用的垃圾收集器非常rudimentary,因此不适合严肃使用。 所以我想知道:为什么不只将Scala转换为Go (即Scala.js)?这将是
最近,我一直在研究docker及其对SaaS公司的有用性。我花了一些时间学习如何容器化应用程序,并简要了解了什么是docker和容器。我在理解这项技术的实用性时遇到了一些问题。我看过dockercon
我必须根据出现在它们之前的字符串是否是某个关键字“load”从输入文件中读取整数。没有关键数字告诉我们要输入多少个数字。这些数字必须保存到一个数组中。为了避免为扫描的每个附加数字创建和更新新数组,我想
Deferred 对象具有回调池 doneCallbacks、failCallbacks 和 progressCallbacks。 doneCallbacks 和 failCallbacks(以及方法
这个问题在这里已经有了答案: Is there a case where including the same header twice is actually helpful? (6 个答案) 关
我在C++ Programming Language的书上看到了下面的例子 class Ptr { X* operator->( ); }; voide f(Ptr p) { p-
你能不能: template const T &operator[] (unsigned int x) 我的想法是如果你有一个 map如果有一个包装器类可以让您这样做,那就太好了: obj["Int
根据doc这个tutorial , cmp() returns -1 if x y 教程里也说了 cmp() returns the sign of the difference of two nu
我经常读到 It seem that identity monad is useless. It's not... but that's another topic. 那么谁能告诉我它有什么用? 最佳
我已经知道实现和接口(interface)的基础知识。我不明白什么时候使用接口(interface)。有接口(interface)的要求是什么? 例子: /// Interface demo Inte
在一些 R 函数的主体中,例如 lm,我看到对 match.call 函数的调用。正如其帮助页面所述,当在函数内部使用 match.call 时,会返回指定参数名称的调用;这对于将大量参数传递给另一个
在监督学习中,我有典型的训练/测试分割来学习算法,例如回归或分类。关于无监督学习,我的问题是:训练/测试分割是否必要且有用?如果是,为什么? 最佳答案 这取决于问题、数据集的形式以及用于解决特定问题的
我最近接触到 Javascript 模板并变得非常感兴趣。 我正在使用 MVC 模式构建一个大型 PHP 应用程序。模板由相当棒的 Twig 处理. 我最近遇到了一个 javascript imple
我最近在一个我要重构并拥有的项目中遇到了以下代码行: SomeClass someClass = new SomeClass(); 我这辈子都想不通为什么有人会以这种方式使用泛型。我想出的唯一原因是
亲爱的,我正在阅读这篇关于通过 asp.net 4 中的代码动态添加元标记的帖子 - 但我想问一下对 SEO 有什么好处,静态添加它或者在代码后面添加它没有问题 http://weblogs.asp.
我是一名优秀的程序员,十分优秀!