讲下背景,目前项目当中使用occi的一个封装库编写dao,但这个方案并不是特别理想。
对于Oracle对待occi的态度一直不太理解,oci的api非常复杂,而对简单很多的occi Oracle支持力度却非常有限。
在Oracle安装版本当中,只提供很少的几个occi编译库。
而且occi库对编译器,编译器版本,数据库版本,crt版本,操作平台版本都有诸多限制。
尽管window上occi也提供了支持vs的链接库,但在window上并不是任何版本的vs都能支持,稍老的05都不支持10G带的的occi库。
在linux上甚至有低版本gcc支持,然后版本高点的gcc反而不支持的情况。
以前写代码本来期望window上mingw测试,unix上部署,结果发觉mingw根本没有能用的occi链接库,而在unix上都得很折腾然后才能让occi正常跑起来。
要注意的是occi支持的编译器、OS平台、各自版本都有限制,如果通过occi访问Oracle一定要确认未来的production运行和编译环境是否明确在支持列表。
采取oci访问Oracle则好很多,c库在mingw,vs都可以透明使用,而且对各个主流OS和编译器也都能很好支持,这样很容易就做到一次编写,多处运行,window下用mingw写出的代码很容易移到unix或linux的gcc甚至acc下。
但oci的缺点也很明显:api太复杂,一个简单的功能也要写大量代码。
于是出现了一些封装oci但是是以类jdbc或者occi风格的wrapper库,比如oraLib,ocilib,ocl。这里介绍一下ocilib:
ocilib封装的oci.lib,以c lib的形式提供给用户,这样使得基于它编写的代码如果今后转到其他平台和编译器上几乎没有需要改动的地方。
并且ocilib支持的数据库特性非常多,换句话说它对多数的oci api都提供了封装,并且以简洁的接口暴露给开发者。
ocilib是lgpl的license,这意味你只要不静态链接ocilib或者封装它,可以免费用到你的商业产品中(封装的话需要开源封装部分的代码)。
下面是一个简单的建立数据库连接的代码:
#include "ocilibDemo.h"
void err_handler(OCI_Error *err)
{
printf(
"code : ORA-%05i\n"
"msg : %s\n"
"sql : %s\n",
OCI_ErrorGetOCICode(err),
OCI_ErrorGetString(err),
OCI_GetSql(OCI_ErrorGetStatement(err))
);
}
void conn()
{
OCI_Connection *cn;
if (!OCI_Initialize(err_handler, NULL, OCI_ENV_DEFAULT))
return;
cn = OCI_ConnectionCreate(tnsName, user, password, OCI_SESSION_DEFAULT);
if (cn != NULL) {
printf(OCI_GetVersionServer(cn));
printf("Server major version : %i\n", OCI_GetServerMajorVersion(cn));
printf("Server minor version : %i\n", OCI_GetServerMinorVersion(cn));
printf("Server revision version : %i\n", OCI_GetServerRevisionVersion(cn));
printf("Connection version : %i\n", OCI_GetVersionConnection(cn));
/* ... application code here ... */
OCI_ConnectionFree(cn);
}
OCI_Cleanup();
return;
}
其中err_handler是作为异常处理函数,会在数据库error发生时被调用。
注意ocilib代码在编译时要设置宏/DOCI_API=__stdcall和/DOCI_CHARSET_XXX,XXX是具体字符集类型有三种:
- OCI_CHARSET_ANSI : all strings are in ANSI
- OCI_CHARSET_UNICODE : all strings are Unicode (versions of Oracle> = 9i)
- OCI_CHARSET_MIXED : SQL statements + metadata are in ANSI, user data and resultset data in Unicode
客户端和服务器端最好都统一设置ANSI或者UNICODE,避免麻烦。
下面是些简单的dml代码:
void createTable(){
OCI_Statement *st = NULL;
OCI_Connection *cn;
if (!OCI_Initialize(err_handler, NULL, OCI_ENV_DEFAULT))
return;
cn = OCI_ConnectionCreate(tnsName, user, password, OCI_SESSION_DEFAULT);
if (cn!=NULL) {
st = OCI_StatementCreate(cn);
char sql[]=MT("create table test_table ")
MT("( ")
MT("val_int number, ")
MT("val_flt float, ")
MT("val_str varchar2(30), ")
MT("val_date date")
MT(")");
cout<<"Create table:\n"<<sql<<endl;
OCI_ExecuteStmt(st, sql);
OCI_ConnectionFree(cn);
}
OCI_Cleanup();
}
void dropTable(){
OCI_Statement *st = NULL;
OCI_Connection *cn;
if (!OCI_Initialize(err_handler, NULL, OCI_ENV_DEFAULT))
return;
cn = OCI_ConnectionCreate(tnsName, user, password, OCI_SESSION_DEFAULT);
if (cn!=NULL) {
st = OCI_StatementCreate(cn);
char sql[]=MT("drop table test_table");
cout<<"Drop table:\n"<<sql<<endl;
OCI_ExecuteStmt(st, sql);
OCI_ConnectionFree(cn);
}
OCI_Cleanup();
}
void insertBind(){
OCI_Date *date;
int i;
double flt;
OCI_Statement *st = NULL;
OCI_Connection *cn;
if (!OCI_Initialize(err_handler, NULL, OCI_ENV_DEFAULT))
return;
cn = OCI_ConnectionCreate(tnsName, user, password, OCI_SESSION_DEFAULT);
if (cn!=NULL) {
st = OCI_StatementCreate(cn);
char sql[]=MT("insert into test_table ")
MT("( ")
MT(" val_int, val_flt, val_str, val_date")
MT( ") " )
MT( "values ")
MT( "( ")
MT( " :val_int, :val_flt, :val_str, :val_date")
MT(") ");
cout<<"Intsert table:\n"<<sql<<endl;
OCI_Prepare(st, sql);
i = 1;
flt = 3.14;
string s="sfsdfsdfsfsdfsdfsd";
date = OCI_DateCreate(cn);
OCI_DateSysDate(date);
OCI_BindInt(st, MT(":val_int"), &i);
OCI_BindDouble(st, MT(":val_flt"), &flt);
OCI_BindString(st, MT(":val_str"), const_cast<char *>(s.c_str()), 30);
OCI_BindDate(st, MT(":val_date"), date);
OCI_Execute(st);
OCI_DateFree(date);
OCI_Commit(cn);
OCI_ConnectionFree(cn);
}
OCI_Cleanup();
}
void insertArray(){
OCI_Connection *cn;
OCI_Statement *st;
OCI_Error *err;
int count=20000000;
int batchSize=5000;
int tab_int[batchSize];
double tab_flt[batchSize];
char tab_str[batchSize][31];
OCI_Date* tab_date[batchSize];
int i;
if (!OCI_Initialize(err_handler, NULL, OCI_ENV_DEFAULT))
return;
cn = OCI_ConnectionCreate(tnsName, user, password, OCI_SESSION_DEFAULT);
st = OCI_StatementCreate(cn);
OCI_Prepare(st, "insert into test_table values(:i, :f, :s, :t)");
OCI_BindArraySetSize(st, batchSize);
OCI_BindArrayOfInts(st, ":i", tab_int, 0);
OCI_BindArrayOfDoubles(st, ":f", tab_flt, 0);
OCI_BindArrayOfStrings(st, ":s", (char*) tab_str, 30, 0);
OCI_BindArrayOfDates(st,":t",tab_date,0);
for (i=0;i<batchSize;i++) {
tab_int[i] = i+1;
tab_flt[i]=3.14;
sprintf(tab_str[i],"Name %d",i+1);
tab_date[i] = OCI_DateCreate(cn);
OCI_DateSysDate(tab_date[i]);
}
int round=count/batchSize;
clock_t start=clock();
cout<<start<<endl;
for (int j=0;j<round;j++) {
if (!OCI_Execute(st)) {
printf("Number of DML array errors : %d\n", OCI_GetBatchErrorCount(st));
err = OCI_GetBatchError(st);
while (err) {
printf("Error at row %d : %s\n", OCI_ErrorGetRow(err), OCI_ErrorGetString(err));
err = OCI_GetBatchError(st);
}
}
OCI_Commit(cn);
// printf("row processed : %d\n", OCI_GetAffectedRows(st));
}
clock_t stop=clock();
cout<<stop<<endl;
int costTime=stop-start;
double numberPerSec=(count/costTime)*1000;
cout<<"Insert records "<<count<<" cost time "<<costTime<<" ms"<<endl;
cout<<"Insert records "<<numberPerSec<<"records/s "<<endl;
for (i=0;i<batchSize;i++) {
OCI_DateFree(tab_date[i]);
}
OCI_Commit(cn);
OCI_ConnectionFree(cn);
OCI_Cleanup();
return;
}
ocilib的代码可以跑在装有Oracle client或instant client的任何机器上,通过tns访问数据库,如果是instant client注意设置tns和home路径。
更多的例子可以查看ocilib自带的demo和文档,ocilib的demo非常完备,提供了它支持的所有特性的例子
分享到:
相关推荐
oci库是Oracle数据库的核心组件之一,它提供了低级别的数据访问功能,包括连接管理、游标操作、SQL执行、事务控制等。oci库允许程序员编写直接与Oracle服务器通信的应用程序,而无需依赖中间件或ODBC等抽象层。 **...
总结来说,“基于OCI的跨平台的Oracle数据库读取类”是一个经过实践检验的解决方案,它利用ocilib库在oci的基础上进行了封装,旨在简化Oracle数据库访问,提高开发效率,同时具备跨平台能力,适应包括Linux在内的...
ocilib 3.8.1 for Windows是开发人员访问Oracle数据库的理想工具,它简化了OCI API,提供了易于理解和使用的接口,同时保持了性能和功能的强大。无论你是新手还是经验丰富的开发者,ocilib都能帮助你更高效地实现...
ocilib是一个开源的C++库,专为访问Oracle数据库设计,它提供了对Oracle Call Interface (OCI) 的高级封装。这个库的主要目的是简化开发人员在C++环境中与Oracle数据库进行交互的工作,使得操作更加方便、高效。 **...
总之,这个"OCI的C++封装库(VS2010)"为C++开发者提供了一个方便、高效且易用的工具,帮助他们在VS2010环境中轻松地进行Oracle数据库的开发,同时保持良好的跨平台兼容性。开发者只需关注业务逻辑,无需过多关心...
OCILIB 是一个第三方库,它作为 OCI 的一个封装,使得开发者可以更方便、高效地利用 OCI 功能,而无需直接处理底层细节。 OCILIB 提供了以下关键特性: 1. **易用性**:OCILIB 通过简化和抽象化 OCI 的复杂 API,...
OCILIB是一款强大的Oracle数据库访问库,它提供了一种简单且高效的API,使得开发者能够用C语言或Delphi编程语言与Oracle数据库进行交互。这个压缩包"OCILIB_20140529_1835 delphi VC"显然是OCILIB的一个特定版本,...
OCILIB是一款开源的C语言库,它为开发者提供了一种简单、高效的方式来访问Oracle数据库。OCILIB封装了Oracle Call Interface (OCI),这是Oracle官方提供的用于与Oracle数据库进行交互的API。通过使用OCILIB,开发者...
oci Linux的开发包主要关注的是Oracle Call Interface (OCI) 的C++封装库,这使得C++开发者能够方便地与Oracle数据库进行交互。oci Linux提供的工具和库为在Linux环境中使用Oracle数据库提供了强大的支持。以下是对...
OCILIB是一款为Oracle数据库提供的C语言接口,它对Oracle的C语言接口OCI进行了封装,使得开发人员能够更加方便和高效地在C/C++环境中编程,开发Oracle数据库的应用程序。OCILIB不仅简化了数据库连接和操作的复杂性,...
**ocilib-3.9.3-gnu.tar.gz_ocilib** 是一个开源的C语言客户端库,专门用于封装Oracle数据库的OCI(Oracle Call Interface)函数。这个库的主要目标是简化开发者与Oracle数据库之间的交互,减轻使用原始OCI接口时的...
在描述中提到,由于Qt本身并不直接支持Oracle接口,因此我们可以采用C语言编写动态链接库(DLL),封装Oracle的OCI接口,然后在Qt应用中调用这个DLL。这种方式使得我们可以独立于Qt平台特性来实现数据库操作,确保...
在VC++环境下进行OCI编程,我们可以利用Microsoft Visual C++的开发工具,结合OCI库来实现对Oracle数据库的高效访问和操作。下面将详细探讨如何在VC++环境中设置和使用OCI进行数据库的链接、查询以及其他数据库操作...
然而,不同封装库的实现可能会有所不同,如ocilib虽然提供了多行fetch的功能,但在某些版本的示例中可能并未明确展示。在这种情况下,深入阅读用户指南和API文档是解决问题的关键,以便更好地理解和利用这些功能。