PostgreSQL作为目前最优秀的跨平台的开源数据库,其介绍及优点这里不再赘述,它能让我以最快的速度忘记my*,ms*。libpq是PostgreSQL的C应用程序接口,它允许客户程序向PostgreSQL服务器进程发送查询并获得查询结果。

我的实验环境是:

操作系统:Linux wheezy 3.2.0-4-amd64 #1 SMP Debian 3.2.41-2 x86_64 GNU/Linux
数据库:EnterpriseDB (PostgreSQL Plus Advanced Server)9.2.1.3 
示例语言:c++
工具:cmake2.8.9,g++ (Debian 4.7.2-5) 4.7.2
 
可能大家只知道PostgreSQL,对EnterpriseDB则比较陌生,这里简单介绍一下EnterpriseDB。EnterpriseDB是在PostgreSQL基础上,针对企业级应用进行了专门的优化,同时增加了一系列如动态性能调优(DynaTune)、EDB Loader、高效批量SQL处理等高级特性;另外EnterpriseDB与Oracle近乎完美的兼容性以及相对Oracle低廉的价格更是一大亮点,这些优异的特点加上IBM与Redhat的巨额注资,我看好它。EnterpriseDB的libpq与PostgreSQL开源版本完全一致,所以这里您也可以使用PostgreSQL。
 
一、准备项目文件CMakeLists.txt,设置libpq的依赖项
使用 libpq 的前端程序必须包括头文件 libpq-fe.h 并且必须与 libpq 库链接。我们创建一个简单的示例项目,并且用cmake作为项目管理工具,这样可以同时复习之前介绍过的cmake相关的知识。
 
创建CMakeLists.txt文件,里面的内容如下:
 
 
    
  1. project(edbconn) 
  2. cmake_minimum_required(VERSION 2.8) 
  3. aux_source_directory(. SRC_LIST) 
  4. add_executable(${PROJECT_NAME} ${SRC_LIST}) 
  5.  
  6. SET(PG_BASE "/opt/PostgresPlus/9.2AS"
  7. INCLUDE_DIRECTORIES("${PG_BASE}/include"
  8. LINK_DIRECTORIES("${PG_BASE}/lib"
  9. TARGET_LINK_LIBRARIES(${PROJECT_NAME} "${PG_BASE}/lib/libpq.so"
 
简单解释一下:
INCLUDE_DIRECTORIES("${PG_BASE}/include")
告诉编译器在哪里去找PostgreSQL的头文件,我的EnterpriseDB安装目录是在/opt/PostgresPlus/9.2AS,INCLUDE_DIRECTORIES("/opt/PostgresPlus/9.2AS/include")这样写也是一样的。
LINK_DIRECTORIES("/opt/PostgresPlus/9.2AS/lib")
告诉编译器外部依赖库,也就是libpq的位置
最后不要忘了告诉编译器依赖项
TARGET_LINK_LIBRARIES(${PROJECT_NAME} "/opt/PostgresPlus/9.2AS/lib/libpq.so")
 
二、用c++通过libpq连接PostgreSQL数据库
类似学习一门新的编程语言从hello world!开始,现在先从连接数据库开始。这次demo是用c++作为开发语言,我陆续会写写 C#与PostgreSQL相关的笔记。
 
别忘记包含libpq-fe.h头文件 。
使用libpq连接PostgreSQL,首先要创建一个PGconn对象。
PGconn *conn;
1、PQconnectdb
定义:PGconn *PQconnectdb(const char *conninfo);
函数PQconnectdb的参数conninfo定义为:
 
 
    
  1. char *conninfo = "hostaddr=127.0.0.1 port=5444 dbname=mydb user=enterprisedb password=edb"
conninfo字符串指定数据库的连接信息,各参数之间用空格隔开。参数hostaddr指定数据库主机的IP地址,port指定EnterpriseDB的端口,dbname指定连接的数据库,user和password...。
conn = PQconnectdb(conninfo); 
2、PQsetlogin
定义:PGconn *PQsetdbLogin(const char *pghost,
             const char *pgport,
             const char *pgoptions,
             const char *pgtty,
             const char *dbName,
             const char *login,
             const char *pwd);
如果没有给固定参数指定默认值,写NULL或一个空字符串就可以了。
 
 
    
  1. conn = PQsetdbLogin("localhost","5444",NULL,NULL,"mydb","enterprisedb","edb"); 
PQconnectdb和PQsetdbLogin函数区别不大,怎么用方便就怎么用。
 
三、连接状态函数
1、返回连接建立时的参数,这些值是固定的
返回连接的数据库名 
char *PQdb(const PGconn *conn);
返回连接的用户名
char *PQuser(const PGconn *conn);
返回连接的密码
char *PQpass(const PGconn *conn);
返回连接的服务器主机名
char *PQhost(const PGconn *conn);
返回连接的端口
char *PQport(const PGconn *conn);
2、返回对PGconn对象操作时的状态数据 
返回连接的状态。
ConnStatusType PQstatus(const PGconn *conn);
ConnStatusType的值最常用的两个是CONNECTION_OK或 CONNECTION_BAD。
 
 
    
  1. // 判断与PostgreSQL的连接是否成功 
  2. if (PQstatus(conn) != CONNECTION_OK) 
  3.     // do nothing... 
查询服务器当前的参数设置
const char *PQparameterStatus(const PGconn *conn, const char *paramName);
参数paramName可使用的值有:server_version , server_encoding, client_encoding, session_authorization, DateStyle, TimeZone, integer_datetimes 和 standard_conforming_strings
 
 
    
  1. // PQparameterStatus 
  2. //  
  3. cout << "server_version : " << PQparameterStatus(conn,"server_version") << endl; 
  4. cout << "server_encoding : " << PQparameterStatus(conn,"server_encoding") << endl; 
  5. cout << "client_encoding : " << PQparameterStatus(conn,"client_encoding") << endl; 
  6. cout << "session_authorization : " << PQparameterStatus(conn,"session_authorization") << endl; 
  7. cout << "DateStyle : " << PQparameterStatus(conn,"DateStyle") << endl; 
  8. cout << "TimeZone : " << PQparameterStatus(conn,"TimeZone") << endl; 
  9. cout << "integer_datetimes : " << PQparameterStatus(conn,"integer_datetimes") << endl; 
  10. cout << "standard_conforming_strings : " << PQparameterStatus(conn,"standard_conforming_strings") << endl; 
 
返回连接中操作产生的最近的错误信息。
char *PQerrorMessage(const PGconn *conn);
 
返回当前连接的进程ID(PID)
int PQbackendPID(const PGconn *conn);
 
到这里,已经可以通过libpq连接到PostgreSQL数据库了。
 
 
    
  1. #include  
  2.  
  3. #include "libpq-fe.h" 
  4.  
  5. using namespace std; 
  6.  
  7. static void exit_nicely(PGconn *conn) 
  8.     PQfinish(conn); 
  9.  
  10. int main() 
  11.         PGconn     *conn; 
  12.  
  13.         /* 
  14.          * PGconn *PQconnectdb(const char *conninfo); 
  15.         */ 
  16.  
  17.         //const char *conninfo = "hostaddr=127.0.0.1 port=5444 dbname=mydb user=enterprisedb password=edb"; 
  18.         // conn = PQconnectdb(conninfo); 
  19.         conn = PQconnectdb("hostaddr=127.0.0.1 port=5444 dbname=mydb user=enterprisedb password=edb"); 
  20.  
  21.         /* 
  22.          *PGconn *PQsetdbLogin(const char *pghost, 
  23.              const char *pgport, 
  24.              const char *pgoptions, 
  25.              const char *pgtty, 
  26.              const char *dbName, 
  27.              const char *login, 
  28.              const char *pwd); 
  29.         */ 
  30.  
  31.         /* Make a connection to the database */ 
  32.         //conn = PQsetdbLogin("localhost","5444",NULL,NULL,"mydb","enterprisedb","edb"); 
  33.  
  34.         /* Check to see that the backend connection was successfully made */ 
  35.         if (PQstatus(conn) != CONNECTION_OK) 
  36.         { 
  37.             cout << "connect failed. PQstatus : " << PQstatus(conn)  << endl; 
  38.             // char *PQerrorMessage(const PGconn *conn); 
  39.             cout << PQerrorMessage(conn) << endl; 
  40.             exit_nicely(conn); 
  41.  
  42.             return 0; 
  43.         } 
  44.  
  45.         // PQdb 
  46.         cout << "数据库名 : " << PQdb(conn) << endl; 
  47.         // PQuser 
  48.         cout << "用户名 : " << PQuser(conn) << endl; 
  49.         // PQpass 
  50.         cout << "口令 : " << PQpass(conn) << endl; 
  51.  
  52.         switch(PQstatus(conn)) 
  53.         { 
  54.         case CONNECTION_STARTED: 
  55.             cout << "正在连接..." << endl; 
  56.             break
  57.  
  58.         case CONNECTION_MADE: 
  59.             cout << "与服务器连接已建立..." << endl; 
  60.             break
  61.  
  62.         case CONNECTION_AUTH_OK: 
  63.             cout << "auth is ok." << endl; 
  64.             break
  65.  
  66.         case CONNECTION_OK: 
  67.             cout << "connected." << endl; 
  68.             break
  69.  
  70.         default
  71.             cout << "正在连接..." << endl; 
  72.         } 
  73.  
  74.         cout << "---------- 连接状态函数 ---------- " << endl; 
  75.         // 
  76.         // PQparameterStatus 
  77.         // server_version , server_encoding, client_encoding, session_authorization, DateStyle, TimeZone, integer_datetimes 和 standard_conforming_strings 
  78.         cout << "server_version : " << PQparameterStatus(conn,"server_version") << endl; 
  79.         cout << "server_encoding : " << PQparameterStatus(conn,"server_encoding") << endl; 
  80.         cout << "client_encoding : " << PQparameterStatus(conn,"client_encoding") << endl; 
  81.         cout << "session_authorization : " << PQparameterStatus(conn,"session_authorization") << endl; 
  82.         cout << "DateStyle : " << PQparameterStatus(conn,"DateStyle") << endl; 
  83.         cout << "TimeZone : " << PQparameterStatus(conn,"TimeZone") << endl; 
  84.         cout << "integer_datetimes : " << PQparameterStatus(conn,"integer_datetimes") << endl; 
  85.         cout << "standard_conforming_strings : " << PQparameterStatus(conn,"standard_conforming_strings") << endl; 
  86.  
  87.         // PQprotocolVersion 
  88.         cout << "PQprotocolVersion : " << PQprotocolVersion(conn) << endl; 
  89.  
  90.         // PQbackendPID 
  91.         cout << "PQbackendPID : " << PQbackendPID(conn) << endl; 
  92.  
  93.         // PQgetssl 
  94.         cout << "PQgetssl : " << PQgetssl(conn) << endl; 
  95.          
  96.         PQfinish(conn); 
  97.         return 0; 

作者:vincent zhang
出处:http://ode.cnblogs.com http://odevincent.blog.51cto.com
Email:wsaspx#hotmail.com

本作品由vincent zhang创作,采用知识共享署名-非商业性使用-禁止演绎 3.0 中国大陆许可协议进行许可。