设为首页 收藏本站
查看: 903|回复: 0

[经验分享] MongoDB:利用官方驱动改装为EF代码风格的MongoDB.Repository框架 二

[复制链接]

尚未签到

发表于 2015-7-6 06:10:42 | 显示全部楼层 |阅读模式
  本次改动的主要内容是实现MongoDB.Repository对MongoDBRef的支持。
  MongoDB对一对一,一对多,多对多关系的维护,官方推荐文档嵌入方式,反映到模型的设计如下:



    public class Student : Entity
{
public string Name { get; set; }
public int Age { get; set; }
}
public class Teacher : Entity
{
public string Name { get; set; }
public int Age { get; set; }
}
public class Grade : RefEntity
{
public string Name { get; set; }
}   
public class School : Entity
{
public string Name { get; set; }
public List Students { get; set; }
public List Teachers { get; set; }
public MongoDBRef Master { get; set; }
}
  
  该种设计在数据库中的存储格式如下图:

  其中的Students, Teachers, Master都以嵌入在School集合中,而在相应的Student,Teacher集合中,并没有相关的数据。
  如果是多学校,我们要查询系统中所有的学生,或老师,那将是一件非常费力的工作。而我们最终目的应该是在School中存储键值,应该使用MongoDBRef实现,如下:



    public class School : Entity
{
public string Name { get; set; }
public List Students { get; set; }
public List Teachers { get; set; }
public MongoDBRef Master { get; set; }
}
  而这种设计,MongoDBRef的数据结构则过于简单,只有Id, CollectionName,DatabaseName三个字段,如果要得到Master,只能去Teacher集合中再次执行查询操作。
  而本次提交,解决的正是这个问题,使代码可以写成如school.Pick(student.Id).Name的形式。
  主要接口:IRefEntity,IDBRefContainer。
  IRefEntity:为减少数据冗余,并提供一个Update方法。Update方法的逻辑是先保存当前实体,即执行IEntity.Save(),然后对DBRefs中的数据进行保存,接口定义如下:



    public interface IRefEntity : IEntity
{
///
/// list of MongoDBRef
///
List DBRefs { get; set; }
///
/// save IEntity first, then save list of MongoDBRef
///
void Update();
}
  IDBRefContainer接口,为IRefEntity.DBRefs的数据提供一个对应的实体容器,已完成相应的查询、添加、删除操作,接口定义如下:



    public interface IDBRefContainer
{
bool Exists(string id);
bool Exists() where T : IEntity;
bool Exists(Predicate match) where T : IEntity;
T Pick(string id) where T : IEntity;
T Pick(Expression where) where T : IEntity;
List GetAll() where T : IEntity;
List GetMany(Expression where) where T : IEntity;
void Add(T entity) where T : IEntity;
void Add(List entities) where T : IEntity;
int Remove(Expression where) where T : IEntity;
void Remove(T entity) where T : IEntity;
void Remove() where T : IEntity;
int Count() where T : IEntity;
int Count(Expression where) where T : IEntity;
List GetAll();
}
  需要注意的是,尽量使用IEntity.Save()进行保存操作,而减少使用IDBRefEntity.Update()进行更新保存 操作,因为IDBRefEntity.Update()针对的是所有IDBRefContainer中的数据,其执行效率还有待改善和提高。本人也一直在纠结,这个IDBRefEntity.Update()是否需要或者是否应该提供。
  下面给出一个具体的测试用例:



        [TestCase]
public void test()
{
grade = new Grade();
grade.Name = "No1";
foreach (Student student in students)
grade.Add(student);
foreach (Teacher teacher in teachers)
grade.Add(teacher);
grade.Update();
students[2].Name = "NameChanged";
students[2].Save();
var g = MongoEntity.Get(grade.Id);
Assert.AreSame(students[2].Name, grade.Pick(students[2].Id).Name);
Assert.AreNotSame(grade.Pick(students[2].Id).Name, g.Pick(students[2].Id).Name);
Assert.AreEqual(grade.Count(), g.Count());
Assert.AreEqual(grade.Count(), g.Count());
}
  具体请参考源码
  本人深感能力不足,欢迎大家指正、指教。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-83488-1-1.html 上篇帖子: MongoDB分片 下篇帖子: 在Window平台安装MongoDB
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表