项目要用到数据库,本来打算用资料更多,名声更好的MysqL
的。卸载了自带的版本装了MariaDB 10.0
,然后发现没有成熟的C++库可用(只有一个提交不久的libmariadb++),于是想回滚到原来的MysqL5.5,但是悲剧的事情发生了…回滚出问题了。用的系统是Linux Mint14
,不管是purge
还是remove
,然后重装MysqL的时候都会卡死…
于是我直接放弃了MysqL
,用上了据说国内很少用,但是企业中用的不少的PostGresql
。 安装过程不说了,用源里面的资源就行,我现在的版本是9.2。
C++库是libpqxx3-dev
,官网有4.0.1的版本,我自己装了一下,链接的时候有些问题1,推荐还是用源里面的版本。貌似必须装到/usr/
下,直接用sudo make install
装到/usr/local
下的话,下面的例子可能通不过…
PostGresql
和MysqL
的管理方式差得挺远,命令上也没啥相似的(除了sql语句)。常用的几条命令包括:
createdb --直接在shell里面输入,而不是连接到数据库后 createuser psql -d database_name -U username -p --连接数据库 \password --连接后,修改密码 \l --列出所有数据库,也可以直接psql -l \d table_name --描述表格,相当于MysqL中的describe \? --其他命令的帮助
容易看出这货很有微软风格…此外,这货还有个GUI的管理程序,挺好用的,名字叫pgadmin
,也可以从源里面装。注意要使用这货还要装postgresql-contrib
这个包,不然会提示没有装管理工具。
我粗略看了一下pqxx
的指引文件,作者封装的比较好了。比较有特色的是实现了数据库的事务功能,也就是所谓的原子操作
,有点类似C++中的异常安全
。但是作者没给出相关的例子,example里面只有简单的过程式编程的例子,如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
#include <iostream> #include <pqxx/pqxx> using namespace std; int main() { try { pqxx::connection conn("dbname=test user=xiaoming"); if(conn.is_open()) { cout<<"Connect successful!"<<endl; cout<<conn.options()<<endl; } else { cout<<"something wrong ..oops"<<endl; return 1; } pqxx::work w(conn); pqxx::result r=w.exec("SELECT * FORM TEST"); w.commit(); for(auto iter=r.begin();iter!=r.end();++r) { auto seg=iter->begin();seg!=iter->end();++seg) cout<<seg->c_str()<<'\t'; cout<<endl; } } catch(exception &e) { cout<<e.what()<<endl; 1; } 0; } |
g++的编译选项:-lpq -lpqxx -std=c++11
别的都容易理解,注意pqxx::result是一个容器,重载了[]
操作符,返回一个tuple
类,这个tuple
不是C++11标准库里面那个,而是作者自己写的。所以他的表现形式和std::tuple
差别挺大的,倒是有点像std::map
,可以直接通过重载的[]
根据列名访问表格的数据,当然用下标指定第几列也是可以的。如果用迭代器访问,最好使用C++11的语法,不然就typedef吧,超级长的类型名。返回的数据类型是pqxx::field
。这个类里面有很多实用函数,如上面的c_str()
,也可以用as<string>()
来获得。
上面这种过程式编程方法也是可行的,自己写个类打个包就能搞了。不过作者给了一种更好的方案,也就是上面提到的事务,使用这种方式,能够更安全简介。我自己写了一段示例代码,作为测试程序:
#include <exception> #include <functional> #include <pqxx/pqxx> #include <pqxx/transaction> #include <pqxx/transactor> namespace std; class Trans:public pqxx::transactor<>{ public: Trans(){} Trans(const string query): query_(query){} void operator()(transaction<> &t){ results_=t.exec(query_); } void on_commit(){ cout<<"query successful!"<<endl; auto iter=results_.begin();iter!=results_.end();++iter){ auto seg=iter->begin();seg!=iter->end();++seg){ cout<<seg->as<string>()<<'\t'; } cout<<endl; } } void on_doubt(){ cout<<"doubt query"<<endl; } void on_abort(const string msg){ cout<<"query Failed: " <<msg<<endl; } private: string query_; pqxx::result results_; }; int main(){ pqxx::connection conn("dbname=test user=terran"); if(conn.is_open()){ cout<<"Connect successful!"<<endl; cout<<conn.options()<<endl; }else{ cout<<"something wrong ...oops"<<endl; 1; } Trans t("SELECT * FROM test_table"); conn.perform(t); 事务的大概用法如上所示,先继承 |