独自handle一个数据库大程有感
这学期数据库课程,最后的大程是写一个MiniSQL的数据库实现,要求很简单,建删表,建删单值索引,支持主键和unique定义,支持最简单的select,只要支持3个类型:int,float,char(0~255)。最开始,考虑到数据库的运行时确定类型的特点,选择了运行时强大的C#,还能顺便集成进Linq。但是一周后发现C#操作对象二进制结构的能力几乎为0,在写BufferManager的时候也发现完全不能自由的控制对象生命周期,并且IDisposable的实现也过于迷,与强大运行时和linq相比,这些弱项是我无法接受的,随即果断转到C++。
C++的RAII提供了精确的对象生命周期和所有权控制,这使我很愉快地写出了能精确控制每一个内存块的BufferManager,智能指针,尤其是unique_ptr在其中起到了巨大的作用,还顺便了解了C++11更加精细的对象左右值的概念,更加熟悉了对象所有权和所有权转移的界定和控制。C++足够接近底层的指针和内存操作,也使得我能直接将一个对象指针reinterpret_cast成byte*,随后直接用二进制模式写回文件(当然这里面我手动控制了对象不能有指针和堆内存),也可以从文件中读入一段二进制内容,直接将首地址reinterpret_cast成对象指针,然后直接使用。模板在其中也帮了很大的忙。
索引的部分,最开始使用了模板B+树,随后发现数据库需要运行时类型,模板这种编译时静态类型的东西不能满足我的需求,于是让模板继承自公用的基类,擦除模板类型,基类的虚函数参数使用byte*,擦除参数类型,通过虚函数调用找到正确的派生类后再由派生类恢复参数的类型参与运算,这样算是一种应急的walk around,毕竟当时的时间不允许我把模板容器改成运行时动态类型容器,也没有时间再造一个动态类型轮子。在把运行时的值转为静态的模板类型时,费了很大的劲,主要是需要实现类似于
template<typename T>
TreeBase* make_tree(); switch(type)
{
case Int:
return make_tree<int>();
case Float:
return make_tree<float>();
case Char:
switch(size)
{
case 1:
return make_tree<array<char, 1>>();
case 2:
return make_tree<array<char, 2>>();
//.....
case 255:
return make_tree<array<char, 255>>();
}
break;
}
类似这样的把动态的值转为静态类型的东西,最恶心的部分是那个switch(size)的部分,要case 1~255,手写的话肯定恶心的要死,于是我写了个模板来做二分搜索,最多6层函数调用就能找到正确的值。糊了这么多,还是为了填当初写了个静态的模板B+树的坑。B+树debug的过程中,使用了VS Debugger提供的natvis可视化工具,来自定义我自己写的类在监视窗口中的显示方式,这东西确实爽,看东西方便了不少。
模板B+树的实现内部坑更多,当初的设计是既然所有类型都是静态的,那就可以静态的确定一个树的节点的度,并且可以使得节点大小正好撑满4KB空间,实现起来出了坑,因为迷之内存对齐占据了空间,使用(4KB – sizeof(非指针成员))/sizeof(指针大小)的方式确定的树节点,总是会使得大小超过4KB,幸亏提前猜到了可能出现这样的问题,加了static_assert,不然又不知道要debug到哪里去,话说回来,最后用了#pragma pack(1)这种坑爹货,关了内存对齐,勉强绕过了这个问题。下面的坑就是来自树节点本身,由于我要把整个节点直接二进制写入文件,所以至少需要保证节点是POD类型(虽然后面发现这样做其实多此一举,还增加复杂度),这也就导致了节点内不能放任何成员函数,于是我又写了一个包装类,持有一个树节点,然后在这个包装类里写需要的成员函数,由于树操作需要同时读多个节点,每个节点是一个block,为了防止读后面节点时前面要用的节点被回收的情况出现,我设计了在包装类在包装节点时自动给节点加锁的机制,这有些类似于shared_ptr的引用计数,随后包装类的锁计数控制又出了坑,这让我明白原来不写的move ctor编译器是会自动生成,而不是把move操作变为copy,因此这里有点怀疑老版C++代码中如果做了类似引用计数的机制,是否能直接在C++11环境下正常运行。
解释器的部分手写了tokenizer,用了vczh轮子叔在他博客里给tinymoe写tokenizer时手写状态机的方法,语法分析是单纯的递归下降同时解释执行。
剩下的部分就不具备太多的技术含量,只是单纯的业务逻辑,唯一的坑点就是我在这个数据库里写了三份管理字符串到数字映射的东西,竟然没想到用hash来做,失败,失败。
最后的测试和运行阶段,跑了一遍VS自带的性能分析工具,真的找到了限制性能的地方,是在lru算法替换文件块的部分,被替换块的选择出了坑,导致块交换频率过高,拉低了程序性能,优化之后,插入1W条数据的同时检查3个unique键并插入索引,只需要20秒多一点,还是比较满意的。
写这个东西的过程中还是发现了C++标准库有很多缺少的东西的,比如序列化(这个在未来有了static reflection应该会有所改善,当然现在也有很多序列化开源库),比如花式拼接字符串(sstream太慢),比如二进制文件流(boost有个buffer_stream,但是太老),比如运行时确定规模并且存储空间连续的二(多)维数组(new是不能new int[m][n]的,n必须是字面值),还有char*和string转换时的麻烦等。
一个人handle这样一个最终写了5000多行的东西,中间用到了各种自己学到的C++技术,尝试了各种从前没用过的工具比如natvis,profiler,还在debug B+树的时候刷了夜,解锁了一大堆成就,虽然最后数据库的分数不是很高(我猜多半是我期中期末考得太烂了,大程目测分不低),还是感觉很有收获的,说了这么多,最后以一句vczh轮子叔曾经说过的话做结尾吧。
你需要花时间做什么,取决于这个问题是不是够难,是不是刚刚好你可以做出来,再难一点点你就做不出来了。只要你保持这种训练方法长达十年,想不牛逼都难。
独自handle一个数据库大程有感的更多相关文章
- mssqlserver数据导出到另外一个数据库
mssqlserver数据导出到另外一个数据库 准备源数据库,找到想要导出的数据库,右键选择"任务"再选择"导出数据" 设置源数据库信息 3.设置目标数据库,导 ...
- 数据库大作业--由python+flask
这个是项目一来是数据库大作业,另一方面也算是再对falsk和python熟悉下,好久不用会忘很快. 界面相比上一个项目好看很多,不过因为时间紧加上只有我一个人写,所以有很多地方逻辑写的比较繁琐,如果是 ...
- 通过 SQL Server 视图访问另一个数据库服务器表的方法
今天项目经理跑过来对我大吼大叫说什么之前安排让我做一大堆接口为什么没做,我直接火了,之前明明没有这个事情…… 不过事情还要解决,好在两个项目都是用的sqlserver,可以通过跨数据库视图来快速解决问 ...
- 怎么用sql语句查询一个数据库有多少张表
今天在技术群中闲谈时忽然聊到一个问题,那就是当一个数据库中有多张表时怎么快速的获取到表的个数,从而给问询者一个准确的回答. 大家或许会说,这个问题和我们的数据库操作没有太大关系或者不是很挂钩,所以没意 ...
- Druid:一个用于大数据实时处理的开源分布式系统
Druid是一个用于大数据实时查询和分析的高容错.高性能开源分布式系统,旨在快速处理大规模的数据,并能够实现快速查询和分析.尤其是当发生代码部署.机器故障以及其他产品系统遇到宕机等情况时,Druid仍 ...
- 有一个很大的整数list,需要求这个list中所有整数的和,写一个可以充分利用多核CPU的代码,来计算结果(转)
引用 前几天在网上看到一个淘宝的面试题:有一个很大的整数list,需要求这个list中所有整数的和,写一个可以充分利用多核CPU的代码,来计算结果.一:分析题目 从题中可以看到“很大的List”以及“ ...
- 如何用 php 读取一个很大的 excel 文件。
这个程序是用php 读取一个很大的excel文件, 先将 excel 文件保存成csv 文件, 然后利用 迭代器 逐行读取 excel 单元格的值, 拿到值以后 做相应处理,并打印结果. <?p ...
- Druid:一个用于大数据实时处理的开源分布式系统——大数据实时查询和分析的高容错、高性能开源分布式系统
转自:http://www.36dsj.com/archives/28590 Druid 是一个用于大数据实时查询和分析的高容错.高性能开源分布式系统,旨在快速处理大规模的数据,并能够实现快速查询和分 ...
- 大项目之网上书城(八)——数据库大改&添加图书
目录 大项目之网上书城(八)--数据库大改&添加图书 主要改动 1.数据库新增表 代码 2.数据库新增触发器 3.其他对BookService和BookDao的修改 代码 4.addBook. ...
随机推荐
- wampserver 403 禁止访问
解决方法:修改Apache配置文件httpd.conf,注释掉 deny from all:将Allow from 127.0.0.1改为Allow from all
- Linux IO漫谈
本文为原创,转载请注明:http://www.cnblogs.com/gistao/ Background IO可能是我们接触最频繁的系统调用,比如printf到终端,send content到对端, ...
- ionic的tabs
<ion-tabs class="tabs-icon-top/bottom(决定这个tabs是置于上面还是底部) tabs-color-active-positive(图标与字体色) ...
- 小米note3,华为手机,软键盘弹出之后,页面上定位的元素布局会乱掉
原因:可能是因为,软键盘弹出时,改变了height,使height:100%,不能达到原来的高度. 解决办法: $(document).ready(function () { $('body').he ...
- SVN更新报错
将服务器SVN文件更新到本地是出现下图错误 报错中已经提示可以通过clean up来清理,若直接执行release lock,则不会解决问题. 原因:本地的项目中存在过期的工作副本 解决办法:选择该文 ...
- 【转】对MVC、MVP、MVVM的懂得
[转]对MVC.MVP.MVVM的懂得 转载地址:http://www.myexception.cn/vc-mfc/1612241.html 对MVC.MVP.MVVM的理解 最近看了一堆js框架的文 ...
- windows环境下创建 .文件夹
一.windows环境下创建 .文件夹 1.新建一个文件夹 2.重命名为.properties.(名字前后都加点) 二.windows环境下创建 .文件 1.上面的方法对文件同样适用 2.运行CMD, ...
- angularjs compile和link
原文:http://www.cnblogs.com/GoodPingGe/p/4361354.html ************************************************ ...
- unity行为树制作AI简单例子(1)
用行为树来制作AI是非常方便的,今天就给大家简单介绍一下行为树的强大之处. 所用插件 Behavior Designer v1.421 最开始 我使用过Rain插件,不过用过Behavior Desi ...
- 更新日志(建议升级到2016.12.17) && 更新程序的方法
更新程序的方法: 1,在控制面板里点击备份当前数据库文件到磁盘,把当天获取的信息从内存写到磁盘/存储卡.2,下载最新版的源码 wget -O "infopi.zip" " ...