1. 引言

一次偶然的机会接触到了iBoxDB这样一个小型的嵌入式对象数据库。感觉非常惊讶有这样轻巧的数据库。iBoxDB 本身是一个NOSQL 同时也有关系数据库的特点。

说说iBoxDB的优点:

1)无需安装,不像其他数据库比如MongoDB, MySQL 需要安装。iBOXDB只需要某个目录存放最终的数据即可。完全就像操作本地文件一  样,应该说比本地文件更方便。

2) 支持多种设备,只需要添加.Net 或者 Java的驱动程序,就可以在多种设备上使用iBoxD,比如Android, Linux,Windows Phone, PC程序等。

3)提供数据库事务支持,可以使用类SQL语法存取对象数据。

4)提供索引,主键以及主从和多主的数据库热同步。

4) 性能优越,这里展示下iBoxDB官方网站提供的和MongoDB性能测试对比结果

在32位机子下的测试结果,代码见Github

这里可以看到iBoxDB有三种运行模式:File 、 MemoryMappedFile、 InMemory。熟悉操作系统的同学对这三个概念肯定不会陌生。它的性能一个比一个高。

2. iBoxDB 的 CRUD 操作

任何DB的肯定包含CRUD的操作,这里给出一个简单的C#实现。

  1. public class Account
  2. {
  3. public string Name { get; set; }
  4. public int Age { get; set; }
  5. public string Email { get; set; }
  6. public string Desc { get; set; }
  7.  
  8. }
  9.  
  10. public static DB.AutoBox InitAndCreateAutoBox()
  11. {
  12. var dbPath = Path.Combine(Directory.GetCurrentDirectory(), "ibox");
  13. if (!Directory.Exists(dbPath))
  14. {
  15. Directory.CreateDirectory(dbPath);
  16. }
  17.  
  18. var server = new DB(dbPath);
  19. var config = server.GetConfig();
  20.  
  21. config.EnsureTable<Account>("AccountTable", "Name");
  22.  
  23. return server.Open();
  24. }

函数 InitAndCreateAutoBox 初始化iBoxDB的环境,创建需要表以及指定主键,保存最终数据的目录。

从DB.Open()方法可获得一个AutoBox对象,这个对象是操作CRUD的。它不需要管理,也不需要释放。

比如:

  1. static void Main(string[] args)
  2. {
  3. //初始化DB
  4. var box = InitAndCreateAutoBox();
  5.  
  6. //插入数据
  7. box.Insert("AccountTable",
  8. new Account() {Name = "001", Age = 14, Desc = "this is test", Email = "fake@qq.com"});
  9. box.Insert("AccountTable",
  10. new Account() {Name = "002", Age = 88, Desc = "this is test 33", Email = "fake2@qq.com"});
  11. box.Insert("AccountTable",
  12. new Account() { Name = "003", Age = 88, Desc = "this is test 88", Email = "fake3@qq.com" });
  13.  
  14. //更新特定数据
  15. var account = box.SelectKey<Account>("AccountTable", "001");
  16. account.Age = 15;
  17. account.Desc = "fake";
  18. account.Email = "email";
  19. box.Update("AccountTable", account);
  20.  
  21. //删除特定数据
  22. box.Delete("AccountTable", "002");
  23.  
  24. var age15Account = box.Select<Account>("from AccountTable where Age==?", 88);
  25. }

另外一种方式就是利用AutoBox.Cube()获得一个Box对象,这个对象是用于控制事务的。此对象一定要释放,典型的就是利用using语句。

  1. static void Main(string[] args)
  2. {
  3. //初始化DB
  4. var box = InitAndCreateAutoBox();
  5.  
  6. using (var cube = box.Cube())
  7. {
  8. cube.Bind("AccountTable")
  9. .Insert(new Account() {Name = "001", Age = 14, Desc = "this is test", Email = "fake@qq.com"});
  1. cube.Bind("AccountTable")
  1. .Insert(new Account() {Name = "002", Age = 15, Desc = "this is test", Email = "fake@qq.com"});
  1. var result = cube.Commit();
  2. }
  3. }

如果是一次性批量处理数据,最好是在一个事务中进行,建议使用Cube,这样效率可以达到最好。

3. iBoxDB下数据库同步

可以看到iBoxDB对于CRUD的便利,利用iBoxDB我们可以快速的搭建一个数据存储的原型并且Demo。但是如何才能将iBoxDB运用到实际的生产环境肯定还有很多工作需要完成。

如果数据量达到某种程度。对于Website而言数据库服务器很有可能成为网站的性能瓶颈。配置数据库的master slave 实现读写分离已经是老生常谈的问题。

不像MongoDB, 直接可以通过配置实现数据同步,iBoxDB 需要代码来实现同步,它提供两种形式:

不管哪种形式都,iBoxDB都用IBoxRecycler 实现数据同步。

这里自定义一个简单的接口实现,主要就是提取出master中数据存放到内存。

  1. public class Package
  2. {
  3. public Socket Socket;
  4. public byte[] OutBox;
  5. }
  1. public class InMemoryBoxRecycler : IBoxRecycler
  2. {
  3. List<Package> qBuffer;
  4. public InMemoryBoxRecycler(long name, DatabaseConfig config)
  5. {
  6. qBuffer = new List<Package>();
  7. }
  8.  
  9. public void OnReceived(Socket socket, BoxData outBox, bool normal)
  10. {
  11. if (socket.DestAddress == long.MaxValue)
  12. {
  13. return;
  14. }
  15. lock (qBuffer)
  16. {
  17. qBuffer.Add(new Package { Socket = socket, OutBox = outBox.ToBytes() });
  18. }
  19. }
  20. public List<Package> GetPackage()
  21. {
  22. return qBuffer;
  23. }
  1. public void Dispose()
  2. {
  3. qBuffer = null;
  4. }
  5. }

自定义Server来管理所有的master 和 slave 节点

  1. public class MsaterSlaveReplicableServer : LocalDatabaseServer
  2. {
  3. public const int MasterAddress = 10;
  4. public const int SlaveAddress = 11;
  5.  
  6. protected override DatabaseConfig BuildDatabaseConfig(long name)
  7. {
  8. if (name == MasterAddress || name == SlaveAddress)
  9. {
  10. var config = new DBPlatform.Config();
  11. config.EnsureTable<Account>("Account", "Name");
  12. }
  13.  
  14. throw new NotImplementedException(name);
  15. }
  16.  
  17. protected override IBoxRecycler BuildBoxRecycler(long name, DatabaseConfig config)
  18. {
  19. if (name == MasterAddress || name == SlaveAddress)
  20. {
  21. return new InMemoryBoxRecycler(name, config);
  22. }
  23.  
  24. return base.BuildBoxRecycler(name, config);
  25. }
  26. }

最后启动Server:

  1. using (var server = new MsaterSlaveReplicableServer())
  2. {
  3. var master = server.GetInstance(MsaterSlaveReplicableServer.MasterAddress);
  4. var slave = server.GetInstance(MsaterSlaveReplicableServer.SlaveAddress);
  5.  
  6. //在master 节点中插入数据
  7. using (var box = master.Cube())
  8. {
  9. for (var i = 0; i < 10; i++)
  10. {
  11. box.Bind("Account").Insert(
  12. new Account()
  13. {
  14. Name = "account" + i,
  15. Age = i,
  16. Desc = "fake",
  17. Email = "email" + i
  18. });
  19. }
  20. box.Commit();
  21. }
  22.  
  23. //将master数据复制到slave
  24. var recycler = (InMemoryBoxRecycler)master.GetBoxRecycler();
  25. lock (recycler.GetPackage())
  26. {
  27. foreach (var p in recycler.GetPackage())
  28. {
  29. if (p.Socket.SourceAddress == MsaterSlaveReplicableServer.MasterAddress)
  30. {
  31. (new BoxData(p.OutBox)).SlaveReplicate(slave).Assert();
  32. }
  33. }
  34. recycler.GetPackage().Clear();
  35. }
  36.  
  37. //检查slave 中的数据
  38. using (var box = slave.Cube())
  39. {
  40. foreach (var item in box.Select<Account>("from Account", null))
  41. {
  42. Console.WriteLine(item);
  43. }
  44. }
  45. }

4. 总结

iBoxDB 这样轻巧的数据库非常适合做项目Demo,但是在实际的Production环境中还是介意大家用MongoDB 等数据库。

欢迎访问我的个人网站 51zhang.net 网站还在不断开发中…

iBoxDB的学习与使用的更多相关文章

  1. 从直播编程到直播教育:LiveEdu.tv开启多元化的在线学习直播时代

    2015年9月,一个叫Livecoding.tv的网站在互联网上引起了编程界的注意.缘于Pingwest品玩的一位编辑在上网时无意中发现了这个网站,并写了一篇文章<一个比直播睡觉更奇怪的网站:直 ...

  2. Angular2学习笔记(1)

    Angular2学习笔记(1) 1. 写在前面 之前基于Electron写过一个Markdown编辑器.就其功能而言,主要功能已经实现,一些小的不影响使用的功能由于时间关系还没有完成:但就代码而言,之 ...

  3. ABP入门系列(1)——学习Abp框架之实操演练

    作为.Net工地搬砖长工一名,一直致力于挖坑(Bug)填坑(Debug),但技术却不见长进.也曾热情于新技术的学习,憧憬过成为技术大拿.从前端到后端,从bootstrap到javascript,从py ...

  4. 消息队列——RabbitMQ学习笔记

    消息队列--RabbitMQ学习笔记 1. 写在前面 昨天简单学习了一个消息队列项目--RabbitMQ,今天趁热打铁,将学到的东西记录下来. 学习的资料主要是官网给出的6个基本的消息发送/接收模型, ...

  5. js学习笔记:webpack基础入门(一)

    之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...

  6. Unity3d学习 制作地形

    这周学习了如何在unity中制作地形,就是在一个Terrain的对象上盖几座小山,在山底种几棵树,那就讲一下如何完成上述内容. 1.在新键得项目的游戏的Hierarchy目录中新键一个Terrain对 ...

  7. 《Django By Example》第四章 中文 翻译 (个人学习,渣翻)

    书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:祝大家新年快乐,这次带来<D ...

  8. 菜鸟Python学习笔记第一天:关于一些函数库的使用

    2017年1月3日 星期二 大一学习一门新的计算机语言真的很难,有时候连函数拼写出错查错都能查半天,没办法,谁让我英语太渣. 关于计算机语言的学习我想还是从C语言学习开始为好,Python有很多语言的 ...

  9. 多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类)

    前言:刚学习了一段机器学习,最近需要重构一个java项目,又赶过来看java.大多是线程代码,没办法,那时候总觉得多线程是个很难的部分很少用到,所以一直没下决定去啃,那些年留下的坑,总是得自己跳进去填 ...

随机推荐

  1. css 常见时间轴的做法(————————————————时间轴——————————————————)

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  2. 基础才是重中之重~stream和byte[]的概念与转化

    回到目录 多看几篇 之所以写这篇文章完全是因为最近在研究FastDFS这个分布式的文件存储系统,当然这不是我第一次研究它了,就像我们去看一本书,我们不会只看一篇,而是一次次,一篇篇,每看一次会有新的收 ...

  3. 爱上MVC3~为下拉列表框添加一个自定义验证规则

    回到目录 开发它的原因: 之前的同事,也是我的哥们,问我下拉列表框是否可以支持验证,这个问题看似简单,但确实MVC里有为我们提供,所以,只能自己写个扩展了,即自己写一个attribute特性,让它继承 ...

  4. 04- Shell脚本学习--条件控制和循环语句

    条件判断:if语句 语法格式: if [ expression ] then Statement(s) to be executed if expression is true fi 注意:expre ...

  5. struts2学习笔记之八:Action中方法的动态调用

    方法一:action名称+!+方法名称+后缀 Action类中增加addUser()和delUser()方法, package com.djoker.struts2; import org.apach ...

  6. iOS 屏幕适配:autoResizing autoLayout和sizeClass

    1. autoResizing autoresizing是苹果早期的ui布局适配的解决办法,iOS6之前完全可以胜任了,因为苹果手机只有3.5寸的屏幕,在加上手机app很少支持横屏,所以iOS开发者基 ...

  7. Java连接MongoDB进行增删改查

    1.导入必须的包: 详情看项目:http://pan.baidu.com/s/1cvDAOY 2.通过Myeclipse创建WEB项目 3. 3.bean:创建实体类 package com.bean ...

  8. HTML内联元素

    前面的话 用于标记段落里的文本和其他内容组的元素种类很多,本文将这些文本级元素进行简单分类,便于整理和记忆 通用容器 <span>元素是短语内容的通用行内容器,并没有任何特殊语义.可以使用 ...

  9. Cocos2d-x 3.2 学习笔记(十一)Game 2048

    一.规则 游戏2048的规则很简单 2+2=4  4+4=8 8+8=16 16+16=32 ……1024+1024=2048 这游戏挺火的……不过做起来也不难,就用cocos2dx做一下,首先我也没 ...

  10. AsyncTask和Handler对比(转)

    1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操作,并提供接口反馈当前异步执行的程度(可 ...