Chinaunix首页 | 论坛 | 博客
  • 博客访问: 76795
  • 博文数量: 10
  • 博客积分: 2106
  • 博客等级: 大尉
  • 技术积分: 131
  • 用 户 组: 普通用户
  • 注册时间: 2009-02-10 17:04
文章分类

全部博文(10)

文章存档

2014年(1)

2011年(2)

2010年(3)

2009年(4)

我的朋友

分类: C/C++

2010-03-30 10:53:38

    在设计表结构时,往往一个表没有合适的主键,常设计一个自增列,或default uuid列,插入数据时可以不插这个列。这个列往往用于外连接,可是插入后可能就找不到哪个是刚插入的了。此时必须使用带RETURNING子句的插入语句,在插入完成后立即得到键值。
    新的DAU/SDBC系统支持插入操作带RETURNING子句,定义一个或多个返回列。
例子如下:
 
 
 

/**********************************************************
 * test RETURNING
 * usage:
 * $./t_ret sql.cfg
 **********************************************************/

#include

/*****************************************
create table ttype (
        "id" number(2),
        "i64" number(20),
        "i2" number(4),
        "d8" number(12,2),
        "d4" number(4,2),
        "t1" timestamp default systimestamp,
        "r5" raw(128),
        "date1" date default sysdate,
        primary key ("t1")
);

以下是模板,由mkpatt程序生成
*****************************************/
T_PkgType TTYPE_tpl[]={
        {CH_TINY,1,"\"id\" id",0,-1},
        {CH_CNUM,22,"\"i64\" i64"},
        {CH_SHORT,sizeof(short),"\"i2\" integer_para"},
        {CH_DOUBLE,sizeof(double),"\"d8\" d8","%14.2lf"},
        {CH_DOUBLE,sizeof(double),"\"d4\" d4","%6.2lf"},
        {CH_USEC,sizeof(INT64),"\"t1\" t1","YYYY-MM-DD HH24:MI:SS.FF6"},
        {CH_BYTE,128,"\"r5\" r5"},
        {CH_DATE,YEAR_TO_SEC_LEN,"\"date1\" date1",YEAR_TO_SEC},
        {-1,0,"TTYPE","t1|"}
};

typedef struct {
        char id;
        char i64[22];
        short integer_para;
        double d8;
        double d4;
        INT64 t1;
        char r5[128];
        char date1[YEAR_TO_SEC_LEN];
} TTYPE_stu;
int ins_ttype_dao(DAU *DP,char *stmt);//DAO函数


int main(int argc,char *argv[])
{
DAU dau;
char stmt[1024];
TTYPE_stu tt;
int ret;
T_SQL_Connect SQL_Connect;

        if(argc>1) ret=envcfg(argv[1]);
        ret=db_open(&SQL_Connect);
        if(ret) {
                printf("open database error!\n");
                return 1;
        }
        ___SQL_Transaction__(&SQL_Connect,TRANBEGIN);
        DAU_init(&dau,&SQL_Connect,0,&tt,TTYPE_tpl);
//对记录赋值
        tt.id=13;
        tt.integer_para=0;
        tt.d8=123456.3;
        tt.d4=1.2;
        tt.t1=INT64NULL;
        *tt.i64=0;

        ret=ins_ttype_dao(&dau,stmt);
        printf("%s:ret=%d\n",stmt,ret);
        printf("t1=%s,date1=%s\n",rusecstrfmt(stmt,tt.t1,YEAR_TO_USEC),tt.date1);

        tt.id=15;
        ret=ins_ttype_dao(&dau,stmt);

//按最后的timestamp查找:

        tt.id=0;

        *stmt=0;
        ret=prepare_by_PK(&dau,stmt);//公用DAO函数 ,按主键取数据
        printf("prepare %s,ret=%d\n",stmt,ret);
        printf("id=%d,t1=%s,date1=%s\n", tt.id,
                rusecstrfmt(stmt,tt.t1,YEAR_TO_USEC),tt.date1);

        ___SQL_Transaction__(&SQL_Connect,TRANCOMMIT);
        DAU_free(&dau);
        ___SQL_CloseDatabase__(&SQL_Connect);

        return 0;
}

int ins_ttype_dao(DAU *DP,char *stmt)
{
int ret;

    do {//以timestamp为主键,大量用户同时插入可能会重码,。
        if(DP->ins_sth < 0)
                DAU_mk_returning(DP,"t1,date1",stmt); //生成RETURNING子句 t1,date1作为返回列
        ret=DAU_ins_returning(DP,stmt); //插入数据库
   } while (DP->SQL_Connect->Errno == DUPKEY);//如果重码再次插入,总会有不重的时候。
   return ret;
}

 make t_ret
cc -D_GNU_SOURCE -O1 -Wall -I/home/sdbc/sdbc5/sdbc/include   -c -o t_ret.o t_ret.c
cc -o t_ret t_ret.o -L/home/sdbc/sdbc5/sdbc/lib -lDAU -lsqli -lscry -lstr -lsqlora8

sdbc@jgbticket:~/sdbc5/sdbc/dau> ./t_ret sql.cfg
INSERT INTO TICKET.TTYPE ("id","i64","i2","d8","d4","r5") VALUES ( :1, :2, :3, :4, :5, :6) RETURNING TO_CHAR("t1",'YYYY-MM-DD HH24:MI:SS.FF6'), TO_CHAR("date1",'YYYY-MM-DD HH24:MI:SS') INTO :7, :8:ret=0
t1=2010-03-31 09:45:11.517632,date1=2010-03-31 09:45:11
prepare SELECT "id" id,"i64" i64,"i2" integer_para,"d8" d8,"d4" d4,TO_CHAR("t1",'YYYY-MM-DD HH24:MI:SS.FF6') t1,"r5" r5,TO_CHAR("date1",'YYYY-MM-DD HH24:MI:SS') date1 FROM TICKET.TTYPE WHERE "t1"=TO_TIMESTAMP(:1,'YYYY-MM-DD HH24:MI:SS.FF6'),ret=0
id=15,t1=2010-03-31 09:45:11.518613,date1=2010-03-31 09:45:11


你可以看到DAU/SDBC帮你做了什么:
1.通过配置文件处理数据库打开参数,不需要在程序中写,运行时随时可改。
2.自动处理表属主,使用他人建立的表,权限受控。
3.自动生成语句,解决列名大小写及伪列名的问题。有时需要列名大小写敏感,这在ORACLE相当繁琐。
4.自动处理RETURNING子句,解决语句中的绑定变量和RETURNING子句中的绑定变量,日期类型格式处理(这在ORACLE也是挺繁琐的)。
 
阅读(1288) | 评论(1) | 转发(0) |
给主人留下些什么吧!~~

chinaunix网友2011-05-06 10:10:44

你的DUA好像不怎么便捷。我们公司用用JSON + OCI封装了一个数据库操作库。 就提供了几个接口,可以解决所有的数据库操作。 例如: DbQuery( char *sql, map_t param, map_t mOut );