我的BO

1-我的BO之强类型

2-我的BO之数据保护

3-我的BO之状态控制

4-我的BO之导航属性

数据需要导航

数据之间普遍存在关系,做业务处理时往往也是按照关系在数据之间查询和处理。业务处理可看作是各种检查和计算后,保存结果。而检查和计算往往意味着要查数据A,以及数据A相关的数据B,数据B相关的数据C……。

BO 与数据表相对应。一个 BO 类对应一个数据表,一个 BO 对象对应数据表中的一条记录。表间的关系,也是 BO 间的关系。典型的是一对多关系。比如 “买家-订单-订单明细”三者的关系是:一个买家可以拥有多个订单,一个订单对应多个订单明细;反过来,一个订单明细对应一个订单,一个订单对应一个买家。订单是买家的子对象,买家是订单的父对象。

当得到一个订单 BO 时,希望可以通过它的属性得到它的明细,明细是一个 BO 集合,希望可以通过它的另一个属性得到它的买家 BO。同理,当得到一个订单明细 BO,希望可以通过它的一个属性得到它的订单 BO。

通过 BO 的属性获取到另一个 BO 或 BO 的集合,这种属性叫做 BO 的导航属性。

对null值的处理

当父对象可能为null时,导航到父对象的属性有两个,一个是Xxx,另一个是XxxOrNull。比如一个学生一般是对应一个班级,但如果业务逻辑允许新生报到时暂未分配到班级,学生的班级属性就可能为null。为了准确地处理null值的问题,学生的班级对应两个属性:ClassClassOrNull。并规定,Class这个属性不返回null,学生当前没有对应班级时,则抛异常,如果希望返回null而非抛异常,则使用ClassOrNull。当父对象不允许为null时,则不提供XxxOrNull属性。

子对象一般表现为父对象的一个集合属性,若一个对象的子对象数量为零,则规定返回空(Empty)集合,而不是null

class 班级Bo extends BoBase {
public List<学生Bo> get学生List() {
Long id = this.getId();
return boProvider.list学生By班级Id(id); // 见说明1
}
} class 学生Bo extends BoBase {
public 班级Bo get班级() {
Long 班级Id = this.get班级Id();
return boProvider.get班级(班级Id);
}
public 班级Bo get班级OrNull() {
if (this.get班级Id() == null)
return null;
return this.get班级(); // 见说明2
}
}

说明:

  1. boProviderBoBase的属性,一切的 BO 对象都由boProvider产生。
  2. 如果班级Id不是null,却在数据库中找不到该班级数据,则还是会抛异常,也应该抛异常。
  3. 为了便于说明且不影响道理的表达,这里使用了中文命名,实际项目中并非如此。

优点

  • 导航属性让使用者方便地得到 BO 对象的相关对象,屏蔽了内部关系的细节,使用方便。
  • 导航属性包含了找不到时返回null还是抛异常的逻辑,避免了到处编写重复的代码,减少了对null值的处理不当导致空引用异常的风险。
  • 效果:只要得到一个 BO 对象,就可以得到与它相关的全部对象,使业务逻辑代码简洁、安全、灵活。

展望

目前的导航属性是只读的,这点不够面向对象(OO),只是因为实现起来简单。未来若有需要,可以提供修改导航属性同时修改关系的功能,用起来更加OO,可能有一些复杂的技术细节问题要解决。

目前通过导航属性获取(BO)对象,每次都从数据库装载,一定概率下会产生多余的查询。解决方法是在内存中统一缓存,通过ID判断若内存已有则不从数据库中取。这个方案一方面会增加一点内存开销,同时为了满足“就算内存中有,也要从数据库装载”的需求,需要设计复杂一点的机制。还需要解决前面的业务修改了的对象,后面的业务(同一个请求)获取是否应该获取修改后的这个对象版本的问题,我目前觉得应该获取修改后的版本,而不是数据库中那个。无论获取到的是哪个版本,会不会在业务处理上造成Bug,有待进一步研究。

系列导航

1-我的BO之强类型

2-我的BO之数据保护

3-我的BO之状态控制

4-我的BO之导航属性

我的BO之导航属性的更多相关文章

  1. 《Entity Framework 6 Recipes》中文翻译系列 (22) -----第五章 加载实体和导航属性之延迟加载

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 第五章 加载实体和导航属性 实体框架提供了非常棒的建模环境,它允许开发人员可视化地使 ...

  2. 《Entity Framework 6 Recipes》中文翻译系列 (23) -----第五章 加载实体和导航属性之预先加载与Find()方法

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-2  预先加载关联实体 问题 你想在一次数据交互中加载一个实体和与它相关联实体. ...

  3. 《Entity Framework 6 Recipes》中文翻译系列 (25) ------ 第五章 加载实体和导航属性之加载完整的对象图和派生类型上的导航属性

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-5  加载完整的对象图 问题 你有一个包含许多关联实体的模型,你想在一次查询中, ...

  4. 《Entity Framework 6 Recipes》中文翻译系列 (26) ------ 第五章 加载实体和导航属性之延缓加载关联实体和在别的LINQ查询操作中使用Include()方法

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-7  在别的LINQ查询操作中使用Include()方法 问题 你有一个LINQ ...

  5. 《Entity Framework 6 Recipes》中文翻译系列 (27) ------ 第五章 加载实体和导航属性之关联实体过滤、排序、执行聚合操作

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-9  关联实体过滤和排序 问题 你有一实体的实例,你想加载应用了过滤和排序的相关 ...

  6. 《Entity Framework 6 Recipes》中文翻译系列 (28) ------ 第五章 加载实体和导航属性之测试实体是否加载与显式加载关联实体

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-11  测试实体引用或实体集合是否加载 问题 你想测试关联实体或实体集合是否已经 ...

  7. 《Entity Framework 6 Recipes》中文翻译系列 (29) ------ 第五章 加载实体和导航属性之过滤预先加载的实体集合和修改外键关联

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-13  过滤预先加载的实体集合 问题 你想过滤预先加载的实体集合,另外,你想使用 ...

  8. 关系与导航属性(摘自微软MSDN)

    关系与导航属性 本主题概述实体框架如何管理实体间的关系.还对如何映射和操作关系提供了一些指南. 关系.导航属性和外键 在关系数据库中,表之间的关系(也称为关联)是通过外键定义的.外键 (FK) 是用于 ...

  9. EF架构~为导航属性赋值时ToList()的替换方案

    回到目录 今天在进行EF开发时,遇到一个问题,在进行join查询时,类中的一个集合类型的导航属性,在给它赋值时,将查询出来的结果ToList()后,出错了,linq to entity不支持这种操作, ...

随机推荐

  1. 攻防世界(Ctf-Web 新手练习区 Writeup)

    平台地址:adworld.xctf.org.cn 在打着暑假工赚零花钱之余,我将这些题利用空余时间刷了一遍,感觉内心还是比较满足的! 题目:view_source 这道题没啥好说的,在url的前面加上 ...

  2. BLE 广播格式定义

    低功耗蓝牙两类报文 : 广播报文 和 数据报文. 本文讨论广播报文数据段,不包括完整报文其他部分,比如前导,接入地址等 蓝牙设备通过广播表明自己的存在,等待被连接, 就好象一个人站在接口大喊“我要脱单 ...

  3. Vue路由规则中定义参数

    Vue使用routerLinke定义参数的时候  路由规则中不需要更改任何属性. 路由其实就是我们在html中定义的锚点,点击这个连接跳转一个锚点.vue中的路由也是这个原理, 前提是路由必须创建在实 ...

  4. 【leetcode】544. Output Contest Matches

    原题 During the NBA playoffs, we always arrange the rather strong team to play with the rather weak te ...

  5. javascript_07-break 和 continue

    break 和 continue break 立刻退出循环 continue 立即退出当前循环,但退出循环后会从循环的顶部继续执行 //求 200-300 之间的所有的偶数的和,用 continue ...

  6. MySQL增删查改语句(入门)

    目录 create alter: insert delete update select 数据库定义语句: create:创建数据库及表对象 drop:删除数据库及表对象 alter:修改数据库及表对 ...

  7. LTS秘钥协商算法分析

    1.根据RCF文档说法 在1-RTT中有两种密钥协商算法(1-RTT ECDHE和 1-RTT PSK  )和4中0-RTT密钥协商方式(0-RTT PSK, 0-RTT ECDH ,0-RTT EC ...

  8. CISCO设备配置SSH 登陆

    1. 设置设备域名CISCO ip domain name CISCO2.创建密钥Server(config)#crypto key generate rsa The name for the key ...

  9. Loadrunner 计算保留两位小数不四舍五入

    有时候在测试过程中会截取返回值,当你截取的值不是最终的值,需要进行计算后才能使用并且需要保留两位小数,不进行四舍五入的计算: 此时 我使用了各种办法,但是最终我采用了一种最直接,最暴力的方法就是先乘后 ...

  10. ES6 正则扩展

    一.新增 flags 属性 ES6 为正则表达式新增了flags属性,会返回正则表达式的修饰符. // ES5 的 source 属性 // 返回正则表达式的正文 /abc/ig.source // ...