C語言鏈接PostgreSQL數據庫

對不少C/C++程序員來講,操做PostgreSQL數據庫是一個技術難點,本文講述採用freecplus開源框架操做PostgreSQL數據庫,freecplus框架是C語言技術網做者二十年的技術積累,功能強大,簡單易用。mysql

1、源代碼說明

freecplus是一個Linux系統下的C/C++開源框架,源代碼請前往C語言技術網(www.freecplus.net)下載。程序員

本文介紹的是freecplus框架中採用connection和sqlstatement類操做PostgreSQL數據庫。sql

類的聲明文件是freecplus/db/postgresql/_postgresql.h。數據庫

類的定義文件是freecplus/db/postgresql/_postgresql.cpp。數據結構

示例程序位於freecplus/db/postgresql目錄中。oracle

編譯規則文件是freecplus/db/postgresql/makefile。框架

2、概述

本文不會介紹PostgreSQL數據庫、SQL語言和C/C++的基礎知識,您應該是一個職業的C/C++程序員,在閱讀本文以前,您已經掌握了PostgreSQL數據庫和SQL語言的基礎知識。函數

freecplus框架把PostgreSQL提供的庫函數封裝成了connection和sqlstatement類,採用封裝後的類操做PostgreSQL數據庫,代碼簡潔優雅,性能卓越。post

接下來我先列出connection和sqlstatement類的聲明,而後經過流程圖和示例程序介紹它位的用法。性能

3、connection類

PostgreSQL數據庫鏈接connection類的聲明(程序員沒必要關心的私有成員和數據結構未列出):

// PostgreSQL數據庫鏈接池類。
class connection
{
public:
  int m_state;         // 與數據庫的鏈接狀態,0-未鏈接,1-已鏈接。

  CDA_DEF m_cda;       // 數據庫操做的結果或最後一次執行SQL語句的結果。

  char m_sql[10241];   // SQL語句的文本,最長不能超過10240字節。

  connection();        // 構造函數。
 ~connection();        // 析構函數。

  // 登陸數據庫。
  // connstr:數據庫的登陸參數,格式:"host= user= password= dbname= port=",
  // 例如:"host=172.16.0.15 user=qxidc password=qxidcpwd dbname=qxidcdb port=5432"
  // username-登陸的用戶名,password-登陸的密碼,dbname-缺省數據庫,port-mysql服務的端口。
  // charset:數據庫的字符集,如"gbk",必須與數據庫保持一致,不然會出現中文亂碼的狀況。
  // autocommitopt:是否啓用自動提交,0-不啓用,1-啓用,缺省是不啓用。
  // 返回值:0-成功,其它失敗,失敗的代碼在m_cda.rc中,失敗的描述在m_cda.message中。
  int connecttodb(char *connstr,char *charset,unsigned int autocommitopt=0);

  // 提交事務。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  int commit();

  // 回滾事務。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  int  rollback();

  // 斷開與數據庫的鏈接。
  // 注意,斷開與數據庫的鏈接時,所有未提交的事務自動回滾。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  int disconnect();

  // 執行SQL語句。
  // 若是SQL語句不須要綁定輸入和輸出變量(無綁定變量、非查詢語句),能夠直接用此方法執行。
  // 參數說明:這是一個可變參數,用法與printf函數相同。
  // 返回值:0-成功,其它失敗,失敗的代碼在m_cda.rc中,失敗的描述在m_cda.message中,
  // 若是成功的執行了非查詢語句,在m_cda.rpc中保存了本次執行SQL影響記錄的行數。
  // 程序員必須檢查execute方法的返回值。
  // 在connection類中提供了execute方法,是爲了方便程序員,在該方法中,也是用sqlstatement類來完成功能。
  int execute(const char *fmt,...);
};

4、sqlstatement類

PostgreSQL數據庫的SQL語句操做sqlstatement類的聲明(程序員沒必要關心的私有成員和數據結構未列出):

// 操做SQL語句類。
class sqlstatement
{
public:
  int m_state;         // 與數據庫鏈接池的綁定狀態,0-未綁定,1-已綁定。

  char m_sql[10241];   // SQL語句的文本,最長不能超過10240字節。

  CDA_DEF m_cda;       // 執行SQL語句的結果。

  sqlstatement();      // 構造函數。
  sqlstatement(connection *conn);    // 構造函數,同時綁定數據庫鏈接池。
 ~sqlstatement();      // 析構函數。

  // 綁定數據庫鏈接池。
  // conn:數據庫鏈接池connection對象的地址。
  // 返回值:0-成功,其它失敗,只要conn參數是有效的,而且數據庫的遊標資源足夠,connect方法不會返回失敗。
  // 程序員通常沒必要關心connect方法的返回值。
  // 注意,每一個sqlstatement只須要綁定一次,在綁定新的connection前,必須先調用disconnect方法。
  int connect(connection *conn);

  // 取消與數據庫鏈接池的綁定。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  int disconnect();

  // 準備SQL語句。
  // 參數說明:這是一個可變參數,用法與printf函數相同。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  // 注意:若是SQL語句沒有改變,只須要prepare一次就能夠了。
  int prepare(const char *fmt,...);

  // 綁定輸入變量的地址。
  // position:字段的順序,從1開始,必須與prepare方法中的SQL的序號一一對應。
  // value:輸入變量的地址,若是是字符串,內存大小應該是表對應的字段長度加1。
  // len:若是輸入變量的數據類型是字符串,用len指定它的最大長度,建議採用表對應的字段長度。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  // 注意:1)若是SQL語句沒有改變,只須要bindin一次就能夠了,2)綁定輸入變量的總數不能超過256個。
  int bindin(unsigned int position,int    *value);
  int bindin(unsigned int position,long   *value);
  int bindin(unsigned int position,unsigned int  *value);
  int bindin(unsigned int position,unsigned long *value);
  int bindin(unsigned int position,float *value);
  int bindin(unsigned int position,double *value);
  int bindin(unsigned int position,char   *value,unsigned int len);

  // 綁定輸出變量的地址。
  // position:字段的順序,從1開始,與SQL的結果集一一對應。
  // value:輸出變量的地址,若是是字符串,內存大小應該是表對應的字段長度加1。
  // len:若是輸出變量的數據類型是字符串,用len指定它的最大長度,建議採用表對應的字段長度。
  // 返回值:0-成功,其它失敗,程序員通常沒必要關心返回值。
  // 注意:1)若是SQL語句沒有改變,只須要bindout一次就能夠了,2)綁定輸出變量的總數不能超過256個。
  int bindout(unsigned int position,int    *value);
  int bindout(unsigned int position,long   *value);
  int bindout(unsigned int position,unsigned int  *value);
  int bindout(unsigned int position,unsigned long *value);
  int bindout(unsigned int position,float *value);
  int bindout(unsigned int position,double *value);
  int bindout(unsigned int position,char   *value,unsigned int len);

  // 執行SQL語句。
  // 返回值:0-成功,其它失敗,失敗的代碼在m_cda.rc中,失敗的描述在m_cda.message中。
  // 若是成功的執行了非查詢語句,在m_cda.rpc中保存了本次執行SQL影響記錄的行數。
  // 程序員必須檢查execute方法的返回值。
  int execute();

  // 執行SQL語句。
  // 若是SQL語句不須要綁定輸入和輸出變量(無綁定變量、非查詢語句),能夠直接用此方法執行。
  // 參數說明:這是一個可變參數,用法與printf函數相同。
  // 返回值:0-成功,其它失敗,失敗的代碼在m_cda.rc中,失敗的描述在m_cda.message中,
  // 若是成功的執行了非查詢語句,在m_cda.rpc中保存了本次執行SQL影響記錄的行數。
  // 程序員必須檢查execute方法的返回值。
  int execute(const char *fmt,...);

  // 從結果集中獲取一條記錄。
  // 若是執行的SQL語句是查詢語句,調用execute方法後,會產生一個結果集(存放在數據庫的緩衝區中)。
  // next方法從結果集中獲取一條記錄,把字段的值放入已綁定的輸出變量中。
  // 返回值:0-成功,1403-結果集已無記錄,其它-失敗,失敗的代碼在m_cda.rc中,失敗的描述在m_cda.message中。
  // 返回失敗的緣由主要有兩個:1)與數據庫的鏈接已斷開;2)綁定輸出變量的內存過小。
  // 每執行一次next方法,m_cda.rpc的值加1。
  // 程序員必須檢查next方法的返回值。
  int next();
};

5、程序流程

freecplus框架把對PostgreSQL數據庫操做的SQL語句分爲兩種:有結果集的SQL語句和無結果集的SQL語句。

若是SQL語句被執行後,有結果集的產生,稱爲有結果集的SQL,即數據查詢語言DQL,以select關鍵字,各類簡單查詢,鏈接查詢等都屬於DQL。

若是SQL語句被執行後,沒有結果集的產生,稱爲無結果集的SQL,包括數據定義語言DDL(主要是create、drop和alter)和數據操縱語言DML(insert、update和insert)。

也能夠這麼說,查詢的SQL語句會產生結果集,其它的SQL語句不會產生結果集。

一、無結果集SQL的程序的流程

這是一個完程的流程,在實際開發中,若是是執行簡單的SQL語句,第6步和第7步可能不須要,若是SQL語句只執行一次,第7步和第8步之間的循環也不須要。

二、有結果集SQL的程序的流程

這是一個完程的流程,在實際開發中,若是是執行簡單的查詢語句,第6步、第7步和第8步可能不須要,若是結果集中最多隻有一條記錄,第10步和第11步之間的循環也不須要。

6、示例程序

一、建立超女信息表

示例(createtable.cpp)

/*
 *  程序名:createtable.cpp,此程序演示freecplus框架操做PostgreSQL數據庫(建立表)。
 *  做者:C語言技術網(www.freecplus.net) 日期:20190525
*/
#include "_postgresql.h"   // freecplus框架操做PostgreSQL的頭文件。

int main(int argc,char *argv[])
{
  connection conn; // 數據庫鏈接池。

  // 登陸數據庫,返回值:0-成功,其它-失敗。
  // 失敗代碼在conn.m_cda.rc中,失敗描述在conn.m_cda.message中。
  if (conn.connecttodb("host=172.16.0.15 user=postgres password=pwdidc dbname=postgres port=5432","gbk")!=0)
  {
    printf("connect database failed.\n%s\n",conn.m_cda.message); return -1;
  }  

  sqlstatement stmt(&conn); // 操做SQL語句的對象。

  // 準備建立表的SQL語句。
  // 超女表girls,超女編號id,超女姓名name,體重weight,報名時間btime,超女說明memo,超女圖片pic。
  stmt.prepare("\
    create table girls(id    int,\
                       name  varchar(30),\
                       weight   numeric(8,2),\
                       btime timestamp,\
                       memo  text,\
                       pic   bytea,\
                       primary key (id))");
  // prepare方法不須要判斷返回值。

  // 執行SQL語句,必定要判斷返回值,0-成功,其它-失敗。
  // 失敗代碼在stmt.m_cda.rc中,失敗描述在stmt.m_cda.message中。
  if (stmt.execute() != 0)
  {
    printf("stmt.execute() failed.\n%s\n%s\n",stmt.m_sql,stmt.m_cda.message); return -1;
  }

  // 注意,在postgresql數據庫中,建立表也要提交事務,和Oracle、MySQL數據庫不一樣。
  conn.commit();

  printf("create table girls ok.\n");
}

運行效果

在這裏插入圖片描述

二、向超女表中插入5條記錄

示例(inserttable.cpp)

/*
 *  程序名:inserttable.cpp,此程序演示freecplus框架操做PostgreSQL數據庫(向表中插入5條記錄)。
 *  做者:C語言技術網(www.freecplus.net) 日期:20190525
*/
#include "_postgresql.h"   // freecplus框架操做PostgreSQL的頭文件。

// 定義用於超女信息的結構,與表中的字段對應。
struct st_girls
{
  long id;        // 超女編號,用long數據類型對應Oracle無小數的number(10)。
  char name[11];  // 超女姓名,用char[31]對應Oracle的varchar2(30)。
  double weight;  // 超女體重,用double數據類型對應Oracle有小數的number(8,2)。
  char btime[20]; // 報名時間,用char對應Oracle的date,格式:'yyyy-mm-dd hh24:mi:ssi'。
} stgirls;

int main(int argc,char *argv[])
{
  connection conn; // 數據庫鏈接類

  // 登陸數據庫,返回值:0-成功,其它-失敗。
  // 失敗代碼在conn.m_cda.rc中,失敗描述在conn.m_cda.message中。
  if (conn.connecttodb("host=172.16.0.15 user=postgres password=pwdidc dbname=postgres port=5432","gbk")!=0)
  {
    printf("connect database failed.\n%s\n",conn.m_cda.message); return -1;
  }
  
  sqlstatement stmt(&conn); // 操做SQL語句的對象。

  // 準備插入表的SQL語句。
  stmt.prepare("\
    insert into girls(id,name,weight,btime) \
                values(:1,:2,:3,to_date(:4,'yyyy-mm-dd hh24:mi:ss'))");
  // prepare方法不須要判斷返回值。
  // 爲SQL語句綁定輸入變量的地址,bindin方法不須要判斷返回值。
  stmt.bindin(1,&stgirls.id);
  stmt.bindin(2, stgirls.name,10);
  stmt.bindin(3,&stgirls.weight);
  stmt.bindin(4, stgirls.btime,19);

  // 模擬超女數據,向表中插入5條測試信息。
  for (int ii=1;ii<=5;ii++)
  {
    memset(&stgirls,0,sizeof(stgirls)); // 結構體變量初始化。

    // 爲結構體變量的成員賦值。
    stgirls.id=ii;                                 // 超女編號。
    sprintf(stgirls.name,"超女%02d",ii);           // 超女姓名。
    stgirls.weight=ii*2.11;                        // 超女體重。
    strcpy(stgirls.btime,"2018-03-01 12:25:31");   // 報名時間。

    // 執行SQL語句,必定要判斷返回值,0-成功,其它-失敗。
    // 失敗代碼在stmt.m_cda.rc中,失敗描述在stmt.m_cda.message中。
    if (stmt.execute() != 0)
    {
      printf("stmt.execute() failed.\n%s\n%s\n",stmt.m_sql,stmt.m_cda.message); return -1;
    }

    printf("成功插入了%ld條記錄。\n",stmt.m_cda.rpc); // stmt.m_cda.rpc是本次執行SQL影響的記錄數。
  }

  printf("insert table girls ok.\n");

  conn.commit(); // 提交數據庫事務。
}

運行效果

在這裏插入圖片描述

三、更新超女表中的記錄

示例(updatetable.cpp)

/*
 *  程序名:updatetable.cpp,此程序演示freecplus框架操做PostgreSQL數據庫(修改表中的記錄)。
 *  做者:C語言技術網(www.freecplus.net) 日期:20190525
*/
#include "_postgresql.h"   // freecplus框架操做PostgreSQL的頭文件。

int main(int argc,char *argv[])
{
  connection conn; // 數據庫鏈接類

  // 登陸數據庫,返回值:0-成功,其它-失敗。
  // 失敗代碼在conn.m_cda.rc中,失敗描述在conn.m_cda.message中。
  if (conn.connecttodb("host=172.16.0.15 user=postgres password=pwdidc dbname=postgres port=5432","gbk")!=0)
  {
    printf("connect database failed.\n%s\n",conn.m_cda.message); return -1;
  }

  sqlstatement stmt(&conn); // 操做SQL語句的對象。

  char strbtime[20];  // 用於存放超女的報名時間。
  memset(strbtime,0,sizeof(strbtime));
  strcpy(strbtime,"2019-12-20 09:45:30");

  // 準備更新數據的SQL語句,不須要判斷返回值。
  stmt.prepare("\
    update girls set btime=to_date(:1,'yyyy-mm-dd hh24:mi:ss') where id>=2 and id<=4");
  // prepare方法不須要判斷返回值。
  // 爲SQL語句綁定輸入變量的地址,bindin方法不須要判斷返回值。
  stmt.bindin(1,strbtime,19);
  // 若是不採用綁定輸入變量的方法,把strbtime的值直接寫在SQL語句中也是能夠的,以下:
  /*
  stmt.prepare("\
    update girls set btime=to_date('%s','yyyy-mm-dd hh24:mi:ss') where id>=2 and id<=4",strbtime);
  */

  // 執行SQL語句,必定要判斷返回值,0-成功,其它-失敗。
  // 失敗代碼在stmt.m_cda.rc中,失敗描述在stmt.m_cda.message中。
  if (stmt.execute() != 0)
  {
    printf("stmt.execute() failed.\n%s\n%s\n",stmt.m_sql,stmt.m_cda.message); return -1;
  }

  // 請注意,stmt.m_cda.rpc變量很是重要,它保存了SQL被執行後影響的記錄數。
  printf("本次更新了girls表%ld條記錄。\n",stmt.m_cda.rpc);

  // 提交事務
  conn.commit();
}

運行效果
在這裏插入圖片描述

四、查詢超女表中的記錄

示例(selecttable.cpp)

/*
 *  程序名:selecttable.cpp,此程序演示freecplus框架操做PostgreSQL數據庫(查詢表中的記錄)。
 *  做者:C語言技術網(www.freecplus.net) 日期:20190525
*/
#include "_postgresql.h"   // freecplus框架操做PostgreSQL的頭文件。

// 定義用於超女信息的結構,與表中的字段對應。
struct st_girls
{
  long id;        // 超女編號,用long數據類型對應Oracle無小數的number(10)。
  char name[31];  // 超女姓名,用char[31]對應Oracle的varchar2(30)。
  double weight;  // 超女體重,用double數據類型對應Oracle有小數的number(8,2)。
  char btime[20]; // 報名時間,用char對應Oracle的date,格式:'yyyy-mm-dd hh24:mi:ss'。
} stgirls;

int main(int argc,char *argv[])
{
  connection conn; // 數據庫鏈接類
  
  // 登陸數據庫,返回值:0-成功,其它-失敗。
  // 失敗代碼在conn.m_cda.rc中,失敗描述在conn.m_cda.message中。
  if (conn.connecttodb("host=172.16.0.15 user=postgres password=pwdidc dbname=postgres port=5432","gbk")!=0)
  {
    printf("connect database failed.\n%s\n",conn.m_cda.message); return -1;
  }

  sqlstatement stmt(&conn); // 操做SQL語句的對象。

  int iminid,imaxid;  // 查詢條件最小和最大的id。

  // 準備查詢表的SQL語句。
  stmt.prepare("\
    select id,name,weight,to_char(btime,'yyyy-mm-dd hh24:mi:ss') from girls where id>=:1 and id<=:2");
  // prepare方法不須要判斷返回值。
  // 爲SQL語句綁定輸入變量的地址,bindin方法不須要判斷返回值。
  stmt.bindin(1,&iminid);
  stmt.bindin(2,&imaxid);
  // 爲SQL語句綁定輸出變量的地址,bindout方法不須要判斷返回值。
  stmt.bindout(1,&stgirls.id);
  stmt.bindout(2, stgirls.name,30);
  stmt.bindout(3,&stgirls.weight);
  stmt.bindout(4, stgirls.btime,19);

  iminid=2;  // 指定待查詢記錄的最小id的值。
  imaxid=4;  // 指定待查詢記錄的最大id的值。

  // 執行SQL語句,必定要判斷返回值,0-成功,其它-失敗。
  // 失敗代碼在stmt.m_cda.rc中,失敗描述在stmt.m_cda.message中。
  if (stmt.execute() != 0)
  {
    printf("stmt.execute() failed.\n%s\n%s\n",stmt.m_sql,stmt.m_cda.message); return -1;
  }

  // 本程序執行的是查詢語句,執行stmt.execute()後,將會在數據庫的緩衝區中產生一個結果集。
  while (1)
  {
    memset(&stgirls,0,sizeof(stgirls)); // 先把結構體變量初始化。

    // 從結果集中獲取一條記錄,必定要判斷返回值,0-成功,1403-無記錄,其它-失敗。
    // 在實際開發中,除了0和1403,其它的狀況極少出現。
    if (stmt.next() !=0) break;
    
    // 把獲取到的記錄的值打印出來。
    printf("id=%ld,name=%s,weight=%.02f,btime=%s\n",stgirls.id,stgirls.name,stgirls.weight,stgirls.btime);
  }

  // 請注意,stmt.m_cda.rpc變量很是重要,它保存了SQL被執行後影響的記錄數。
  printf("本次查詢了girls表%ld條記錄。\n",stmt.m_cda.rpc);
}

運行效果
在這裏插入圖片描述

五、查詢超女表中的記錄數

示例(counttable.cpp)

/*
 *  程序名:counttable.cpp,此程序演示freecplus框架操做PostgreSQL數據庫(查詢表中的記錄數)。
 *  做者:C語言技術網(www.freecplus.net) 日期:20190525
*/
#include "_postgresql.h"   // freecplus框架操做PostgreSQL的頭文件。

int main(int argc,char *argv[])
{
  connection conn; // 數據庫鏈接類
  
  // 登陸數據庫,返回值:0-成功,其它-失敗。
  // 失敗代碼在conn.m_cda.rc中,失敗描述在conn.m_cda.message中。
  if (conn.connecttodb("host=172.16.0.15 user=postgres password=pwdidc dbname=postgres port=5432","gbk")!=0)
  {
    printf("connect database failed.\n%s\n",conn.m_cda.message); return -1;
  }

  sqlstatement stmt(&conn); // 操做SQL語句的對象。

  int icount=0;  // 用於存放查詢結果的記錄數。

  // 準備查詢表的SQL語句,把查詢條件直接寫在SQL語句中,沒有采用綁定輸入變量的方法。
  stmt.prepare("select count(*) from girls where id>=2 and id<=4");
  // prepare方法不須要判斷返回值。
  // 爲SQL語句綁定輸出變量的地址,bindout方法不須要判斷返回值。
  stmt.bindout(1,&icount);

  // 執行SQL語句,必定要判斷返回值,0-成功,其它-失敗。
  // 失敗代碼在stmt.m_cda.rc中,失敗描述在stmt.m_cda.message中。
  if (stmt.execute() != 0)
  {
    printf("stmt.execute() failed.\n%s\n%s\n",stmt.m_sql,stmt.m_cda.message); return -1;
  }

  // 本程序執行的是查詢語句,執行stmt.execute()後,將會在數據庫的緩衝區中產生一個結果集。
  // 可是,在本程序中,結果集永遠只有一條記錄,調用stmt.next()一次就行,不須要循環。
  stmt.next();
  
  printf("girls表中符合條件的記錄數是%d。\n",icount);
}

運行效果
在這裏插入圖片描述

七、刪除超女表中的記錄

示例(deletetable.cpp)

/*
 *  程序名:deletetable.cpp,此程序演示freecplus框架操做PostgreSQL數據庫(刪除表中的記錄)。
 *  做者:C語言技術網(www.freecplus.net) 日期:20190525
*/
#include "_postgresql.h"   // freecplus框架操做PostgreSQL的頭文件。

int main(int argc,char *argv[])
{
  connection conn; // 數據庫鏈接類

  // 登陸數據庫,返回值:0-成功,其它-失敗。
  // 失敗代碼在conn.m_cda.rc中,失敗描述在conn.m_cda.message中。
  if (conn.connecttodb("host=172.16.0.15 user=postgres password=pwdidc dbname=postgres port=5432","gbk")!=0)
  {
    printf("connect database failed.\n%s\n",conn.m_cda.message); return -1;
  }

  sqlstatement stmt(&conn); // 操做SQL語句的對象。

  // 執行SQL語句,必定要判斷返回值,0-成功,其它-失敗。
  // 失敗代碼在stmt.m_cda.rc中,失敗描述在stmt.m_cda.message中。
  // 若是不須要綁定輸入和輸出變量,用stmt.execute()方法直接執行SQL語句,不須要stmt.prepare()。
  if (stmt.execute("delete from girls where id>=2 and id<=4") != 0)
  {
    printf("stmt.execute() failed.\n%s\n%s\n",stmt.m_sql,stmt.m_cda.message); return -1;
  }

  // 請注意,stmt.m_cda.rpc變量很是重要,它保存了SQL被執行後影響的記錄數。
  printf("本次從girls表中刪除了%ld條記錄。\n",stmt.m_cda.rpc); 

  // 提交事務
  conn.commit();
}

運行效果
在這裏插入圖片描述

八、與Oracle的兼容性處理

在封裝sqlstatement類的時候,爲了與Oracle兼容,作了如下方面的處理:

1)在PostgreSQL中,綁定輸入和輸出變量採用的是"$",Oracle採用的是":n"(n表示變量的序號),在sqlstatement的prepare方法中,把":n"替換成了"$"。

2)在PostgreSQL中,把字符串輸換爲日期時間的函數是to_timestamp,Oracle是to_date,在sqlstatement的prepare方法中,把to_date替換成to_timestamp。

4)PostgreSQL的sqlstatement類綁定輸入或輸出變量的最大數量缺省是256,在"_mysql.h"頭文件中定義了MAXPARAMS宏,您能夠根據實際需求修改它。

// 執行SQL語句前綁定輸入或輸出變量個數的最大值,256是很大的了,能夠根據實際狀況調整。
#define MAXPARAMS  256

5)sqlstatement類綁定輸入或輸出變量時,若是是字符串,最大長度缺省是2000,在"_postgresql.h"頭文件中定義了MAXFIELDLENGTH宏,您能夠根據實際需求修改它。

// 若是綁定輸入或輸出變量是字符串,指定字符串的最大長度,不包括字符串的結束符。
#define MAXFIELDLENGTH 2000

九、text和bytea字段的操做

PostgreSQL提供的庫函數支持對text和bytea字段的操做,本人的技術水平有限,找不到這方面的資料和示例程序,因此尚未封裝對text和bytea字段的操做,但願各位能提供技術幫助,經過C語言技術網與我聯繫,咱們共同完善freecplus框架,很是感謝。

7、應用經驗

本文提供的示例程序看上去簡單,實則很精妙,但願你們多多思考,慢慢體會。

爲了讓你們徹底掌握connection和sqlstatement類的用法,我將錄製freecplus框架的專題視頻,請你們多關注C語言技術網(www.freecplus.net)發佈的內容。

8、版權聲明

C語言技術網原創文章,轉載請說明文章的來源、做者和原文的連接。

來源:C語言技術網(www.freecplus.net

做者:碼農有道

相關文章
相關標籤/搜索