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 = "fake@qq.com"});
box.Insert("AccountTable",
new Account() {Name = "002", Age = 88, Desc = "this is test 33", Email = "fake2@qq.com"});
box.Insert("AccountTable",
new Account() { Name = "003", Age = 88, Desc = "this is test 88", Email = "fake3@qq.com" }); //更新特定数据
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 = "fake@qq.com"});
cube.Bind("AccountTable")
.Insert(new Account() {Name = "002", Age = 15, Desc = "this is test", Email = "fake@qq.com"});
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 网站还在不断开发中…
iBoxDB的学习与使用的更多相关文章
- 从直播编程到直播教育:LiveEdu.tv开启多元化的在线学习直播时代
2015年9月,一个叫Livecoding.tv的网站在互联网上引起了编程界的注意.缘于Pingwest品玩的一位编辑在上网时无意中发现了这个网站,并写了一篇文章<一个比直播睡觉更奇怪的网站:直 ...
- Angular2学习笔记(1)
Angular2学习笔记(1) 1. 写在前面 之前基于Electron写过一个Markdown编辑器.就其功能而言,主要功能已经实现,一些小的不影响使用的功能由于时间关系还没有完成:但就代码而言,之 ...
- ABP入门系列(1)——学习Abp框架之实操演练
作为.Net工地搬砖长工一名,一直致力于挖坑(Bug)填坑(Debug),但技术却不见长进.也曾热情于新技术的学习,憧憬过成为技术大拿.从前端到后端,从bootstrap到javascript,从py ...
- 消息队列——RabbitMQ学习笔记
消息队列--RabbitMQ学习笔记 1. 写在前面 昨天简单学习了一个消息队列项目--RabbitMQ,今天趁热打铁,将学到的东西记录下来. 学习的资料主要是官网给出的6个基本的消息发送/接收模型, ...
- js学习笔记:webpack基础入门(一)
之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...
- Unity3d学习 制作地形
这周学习了如何在unity中制作地形,就是在一个Terrain的对象上盖几座小山,在山底种几棵树,那就讲一下如何完成上述内容. 1.在新键得项目的游戏的Hierarchy目录中新键一个Terrain对 ...
- 《Django By Example》第四章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:祝大家新年快乐,这次带来<D ...
- 菜鸟Python学习笔记第一天:关于一些函数库的使用
2017年1月3日 星期二 大一学习一门新的计算机语言真的很难,有时候连函数拼写出错查错都能查半天,没办法,谁让我英语太渣. 关于计算机语言的学习我想还是从C语言学习开始为好,Python有很多语言的 ...
- 多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类)
前言:刚学习了一段机器学习,最近需要重构一个java项目,又赶过来看java.大多是线程代码,没办法,那时候总觉得多线程是个很难的部分很少用到,所以一直没下决定去啃,那些年留下的坑,总是得自己跳进去填 ...
随机推荐
- Android Fragment add/replace以及backstack
无论Fragment以何种方式加入,都不会影响backstack,backstack由addToBackStack函数决定,只有调用了这个函数,才会将Fragment加入返回栈.这个说法其实不太准确, ...
- Java连接Oracle数据库开发银行管理系统【一、需求篇】
此系统开发共分为三篇完成. 第一篇[需求篇]:效果展示图,也就是需求部分的展示 第二篇[设计篇]:需求分析和类,接口的设计 第三篇[实现篇]:具体代码实现
- 知方可补不足~SQL2005使用ROW_NUMBER() OVER()进行数据分页
回到目录 数据分页是这个经常说的东西,无论在WEBForm还是WinForm中它都会被单独拿出来,或者是公用组件,或者是公用类库,反正对于数据分页这个东西,总是我们关注的一个话题,但事实上,数据分页归 ...
- Atitit 判断判断一张图片是否包含另一张小图片
Atitit 判断判断一张图片是否包含另一张小图片 1. keyword1 2. 模板匹配是在图像中寻找目标的方法之一(切割+图像相似度计算)1 3. 匹配效果2 4. 图片相似度的算法(感知哈希算 ...
- 12.创建一个Point类,有成员变量x,y,方法getX(),setX(),还有一个构造方 法初始化x和y。创建类主类A来测试它。
package java1; public class Point { int x; int y; Point(int x,int y) { this.x = x; this.y = y; } pub ...
- SQL Server的Execute As与连接池结合使用的测试
简介 在SQL Server中,Execute As关键字允许当前账户在特定上下文中以另一个用户或登录名的身份执行SQL语句,比如用户张三有权限访问订单表,用户李四并没有权限访问订单表,那么给 ...
- 【WP8.1开发】认识后台任务
在手机上,使用后台,不像电脑上那么随意,准确地讲嘛,在移动平台上,后台任务都有严格的限制.至于说为什么会有这么多限制,我估计初衷很明显——保证系统的性能不受某个或某几个应用的负面影响:另外就是出于安全 ...
- ab压力测试
原文链接:http://www.orlion.ga/698/ ab是个什么就不说了搞lamp的都会知道.主要看一下结果都是什么意义. ab 的用法是:ab [options] [http://]hos ...
- CSS颜色模式转换器的实现
前面的话 在CSS中,颜色的表示方式主要包括关键字.16进制.RGB模式.RGBA模式.HSL模式.HSLA模式.关于颜色模式的详细信息移步至此.本文就16进制.RGB模式及HSL模式的互相转换进行实 ...
- Spring学习总结(五)——Spring整合MyBatis(Maven+MySQL)一
MyBatis-Spring 会帮助你将 MyBatis 代码无缝地整合到 Spring 中. 使用这个类库中的类, Spring 将会加载必要的MyBatis工厂类和 session 类. 这个类库 ...