iBoxDB的学习与使用

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#实现。

        public class Account
        {
            public string Name { get; set; }
            public int Age { get; set; }
            public string Email { get; set; }
            public string Desc { get; set; }

        }

        public static DB.AutoBox InitAndCreateAutoBox()
        {
            var dbPath = Path.Combine(Directory.GetCurrentDirectory(), "ibox");
            if (!Directory.Exists(dbPath))
            {
                Directory.CreateDirectory(dbPath);
            }

            var server = new DB(dbPath);
            var config = server.GetConfig();

            config.EnsureTable<Account>("AccountTable", "Name");

            return server.Open();
        }

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

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

比如:

        static void Main(string[] args)
        {
            //初始化DB
            var box = InitAndCreateAutoBox();

            //插入数据
              box.Insert("AccountTable",
                new Account() {Name = "001", Age = 14, Desc = "this is test", Email = "[email protected]"});
            box.Insert("AccountTable",
                new Account() {Name = "002", Age = 88, Desc = "this is test 33", Email = "[email protected]"});
            box.Insert("AccountTable",
                new Account() { Name = "003", Age = 88, Desc = "this is test 88", Email = "[email protected]" });

            //更新特定数据
              var account = box.SelectKey<Account>("AccountTable", "001");
            account.Age = 15;
            account.Desc = "fake";
            account.Email = "email";
            box.Update("AccountTable", account);

            //删除特定数据
              box.Delete("AccountTable", "002");

            var age15Account = box.Select<Account>("from AccountTable where Age==?", 88);
        }

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

        static void Main(string[] args)
        {
            //初始化DB
            var box = InitAndCreateAutoBox();

            using (var cube = box.Cube())
            {
                cube.Bind("AccountTable")
                    .Insert(new Account() {Name = "001", Age = 14, Desc = "this is test", Email = "[email protected]"});
                cube.Bind("AccountTable")
                    .Insert(new Account() {Name = "002", Age = 15, Desc = "this is test", Email = "[email protected]"});
                var result = cube.Commit();
            }
        }

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

3. iBoxDB下数据库同步

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

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

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

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

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

    public class Package
    {
        public Socket Socket;
        public byte[] OutBox;
    } 
    public class InMemoryBoxRecycler : IBoxRecycler
    {
        List<Package> qBuffer;
        public InMemoryBoxRecycler(long name, DatabaseConfig config)
        {
            qBuffer = new List<Package>();
        }

        public void OnReceived(Socket socket, BoxData outBox, bool normal)
        {
            if (socket.DestAddress == long.MaxValue)
            {
                 return;
            }
            lock (qBuffer)
            {
                qBuffer.Add(new Package { Socket = socket, OutBox = outBox.ToBytes() });
            }
        }
        public List<Package> GetPackage()
        {
            return qBuffer;
        }
                
        public void Dispose()
        {
            qBuffer = null;
        }
    }

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

    public class MsaterSlaveReplicableServer : LocalDatabaseServer
    {
        public const int MasterAddress = 10;
        public const int SlaveAddress = 11;

        protected override DatabaseConfig BuildDatabaseConfig(long name)
        {
            if (name == MasterAddress || name == SlaveAddress)
            {
                var config = new DBPlatform.Config();
                config.EnsureTable<Account>("Account", "Name");
            }

            throw new NotImplementedException(name);
        }

        protected override IBoxRecycler BuildBoxRecycler(long name, DatabaseConfig config)
        {
            if (name == MasterAddress || name == SlaveAddress)
            {
                return new InMemoryBoxRecycler(name, config);
            }

            return base.BuildBoxRecycler(name, config);
        }
    }

最后启动Server:

              using (var server = new MsaterSlaveReplicableServer())
            {
                var master = server.GetInstance(MsaterSlaveReplicableServer.MasterAddress);
                var slave = server.GetInstance(MsaterSlaveReplicableServer.SlaveAddress);

                //在master 节点中插入数据
                   using (var box = master.Cube())
                {
                    for (var i = 0; i < 10; i++)
                    {
                        box.Bind("Account").Insert(
                            new Account()
                            {
                                Name = "account" + i,
                                Age = i,
                                Desc = "fake",
                                Email = "email" + i
                            });
                    }
                    box.Commit();
                }

                //将master数据复制到slave
                var recycler = (InMemoryBoxRecycler)master.GetBoxRecycler();
                lock (recycler.GetPackage())
                {
                    foreach (var p in recycler.GetPackage())
                    {
                        if (p.Socket.SourceAddress == MsaterSlaveReplicableServer.MasterAddress)
                        {
                            (new BoxData(p.OutBox)).SlaveReplicate(slave).Assert();
                        }
                    }
                    recycler.GetPackage().Clear();
                }

                //检查slave 中的数据
                   using (var box = slave.Cube())
                {
                    foreach (var item in box.Select<Account>("from Account", null))
                    {
                        Console.WriteLine(item);
                    }
                }
            }

4. 总结

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

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

时间: 2024-08-25 17:09:55

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

Vue.js学习笔记:属性绑定 v-bind

v-bind  主要用于属性绑定,Vue官方提供了一个简写方式 :bind,例如: <!-- 完整语法 --> <a v-bind:href="url"></a> <!-- 缩写 --> <a :href="url"></a> 绑定HTML Class 一.对象语法: 我们可以给v-bind:class 一个对象,以动态地切换class.注意:v-bind:class指令可以与普通的class特

Java多线程学习(吐血超详细总结)

林炳文Evankaka原创作品.转载请注明出处http://blog.csdn.net/evankaka 目录(?)[-] 一扩展javalangThread类 二实现javalangRunnable接口 三Thread和Runnable的区别 四线程状态转换 五线程调度 六常用函数说明 使用方式 为什么要用join方法 七常见线程名词解释 八线程同步 九线程数据传递 本文主要讲了java中多线程的使用方法.线程同步.线程数据传递.线程状态及相应的一些线程函数用法.概述等. 首先讲一下进程和线程

微信小程序学习总结(2)------- 之for循环,绑定点击事件

最近公司有小程序的项目,本人有幸参与其中,一个项目做下来感觉受益匪浅,与大家做下分享,欢迎沟通交流互相学习. 先说一下此次项目本人体会较深的几个关键点:微信地图.用户静默授权.用户弹窗授权.微信充值等等. 言归正传,今天分享我遇到的关于wx:for循环绑定数据的一个tips:  1. 想必大家的都知道wx:for,如下就不用我啰嗦了: <view class="myNew" wx:for="{{list}}">{{item.title}}<view

【安全牛学习笔记】

弱点扫描 ╋━━━━━━━━━━━━━━━━━━━━╋ ┃发现弱点                                ┃ ┃发现漏洞                                ┃ ┃  基于端口五福扫描结果版本信息(速度慢)┃ ┃  搜索已公开的漏洞数据库(数量大)      ┃ ┃  使用弱点扫描器实现漏洞管理            ┃ ╋━━━━━━━━━━━━━━━━━━━━╋ [email protected]:~# searchsploit Usage:

winform学习日志(二十三)---------------socket(TCP)发送文件

一:由于在上一个随笔的基础之上拓展的所以直接上代码,客户端: using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.Net.Sockets; using Sys

零基础的人该怎么学习JAVA

对于JAVA有所兴趣但又是零基础的人,该如何学习JAVA呢?对于想要学习开发技术的学子来说找到一个合适自己的培训机构是非常难的事情,在选择的过程中总是 因为这样或那样的问题让你犹豫不决,阻碍你前进的步伐,今天就让小编为您推荐培训机构新起之秀--乐橙谷Java培训机构,助力你成就好未来. 选择java培训就到乐橙谷 北京有什么好的Java培训机构?来乐橙谷北京学Java,零基础走起,乐橙谷Java基础班授课老师经验非常丰富,课程内容安排合理,适合于有一点点Java基础甚至一点都不会Java的同学学

最全解析如何正确学习JavaScript指南,必看!

划重点 鉴于时不时,有同学私信问我:怎么学前端的问题.这里统一回复一下,如下次再遇到问我此问题同学,就直接把本文链接地址发给你了. "前端怎么学"应该因人而异,别人的方法未必适合自己.就说说我的学习方法吧:我把大部分时间放在学习js上了.因为这个js的学习曲线,先平后陡.项目实践和练习啥的,我不说了,主要说下工作之外的时间利用问题.我是怎么学的呢,看书,分析源码.个人这几天统计了一下,前端书籍目前看了50多本吧,大部分都是js的.市面上的书基本,差不多都看过. 第一个问题:看书有啥好处

轻松学习C语言编程的秘诀:总结+灵感

目前在准备一套C语言的学习教程,所以我这里就以C语言编程的学习来讲.注意,讲的是"轻松学习",那种不注重方法,拼命玩命的方式也有其效果,但不是我提倡的.我讲究的是在方式方法对头.适合你.减轻你学习负担和心里压力的前提下,才适当的抓紧时间. 因此,探索一种很好的学习方法就是我所研究的主要内容. 众所周知,学习C语言并非易事,要学好它更是难上加难.这和你期末考试背会几个题目的答案考上满分没多大关系,也就是说你考试满分也说明不了你学好.学精通了C语言.那么怎么才算学精通C语言?闭着眼睛对自己

开始我的Python爬虫学习之路

因为工作需要经常收集一些数据,我就想通过学爬虫来实现自动化完成比较重复的任务. 目前我Python的状况,跟着敲了几个教程,也算是懂点基础,具体比较深入的知识,是打算从做项目中慢慢去了解学习. 我是觉得如果一开始就钻细节的话,是很容易受到打击而放弃的,做点小项目让自己获得点成就感路才更容易更有信心走下去. 反正遇到不懂的就多查多问就对了. 知乎上看了很多关于入门Python爬虫的问答,给自己总结出了大概的学习方向. 基础: HTML&CSS,JOSN,HTTP协议(这些要了解,不太需要精通) R