• postgresql使用手记


    项目要用到数据库,本来打算用资料更多,名声更好的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下的话,下面的例子可能通不过…

    PostGreSQLMySQL的管理方式差得挺远,命令上也没啥相似的(除了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)
                {
                    for(auto seg=iter->begin();seg!=iter->end();++seg)
                        cout<<seg->c_str()<<'\t';
                        cout<<endl;
                }
            }
            catch(exception &e)
            {
                cout<<e.what()<<endl;
                return 1;
            }
    
            return 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>()来获得。

    上面这种过程式编程方法也是可行的,自己写个类打个包就能搞了。不过作者给了一种更好的方案,也就是上面提到的事务,使用这种方式,能够更安全简介。我自己写了一段示例代码,作为测试程序:

    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
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    
    #include <iostream>
    #include <exception>
    #include <functional>
    #include <pqxx/pqxx>
    #include <pqxx/transaction>
    #include <pqxx/transactor>
    
    using 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;
            for(auto iter=results_.begin();iter!=results_.end();++iter){
                for(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;
            return 1;
        }
        Trans t("SELECT * FROM test_table");
        conn.perform(t);
    
        return 0;
    }

    事务的大概用法如上所示,先继承pqxx::transactor<>类(这是个模板,有默认参数),然后实现三个on_函数,把命令放在重载的函数调用操作符里面。调用方只需要简单的.perform就可以了。如果执行成功,会调用on_commit,失败会调用on_abort,提交了但是不能确定是不是成功(一般就是失去连接了),调用on_doubt

    大概就是以上内容吧,pqxx库的实现还是蛮复杂的,其他的还要细看。


    1. 我在编译的时候会提示找不到PQxx的定义,然后编译器提示找不到libpq-fe.h文件,推测是由于安装到/usr/include/postgresql而不是/usr/include的原因。如果有人知道怎么解决,请留言告知,多谢。

  • 相关阅读:
    企业如何才能“勾搭”上服务网格技术?
    行云创新:云原生加速企业释放数据价值
    行云创新:后疫情时代,云原生为酒店数字化转型破局
    行云创新CEO马洪喜荣获“2021杰出质造人物奖”
    SolarMesh发布 v1.6.1版本,再不来体验就......
    什么是云原生?如何建设云原生平台?
    行云创新联合上汽乘用车打造云原生技术平台,加快实现数字化转型
    五分钟搭建你的第一个区块链应用
    mysql 存储过程
    MySQL-binlog日志格式 binlog_format三种模式详解
  • 原文地址:https://www.cnblogs.com/livewithnorest/p/3100160.html
Copyright © 2020-2023  润新知