Oracle Pro*C 动态SQL技术

Pro*C的常用动态SQL技术一共有3种:

  1. 用于处理不包含宿主变量的动态SQL, 不能用于SELECT语句.
  2. 用于处理输入宿主变量个数和类型已经确定的动态SQL, 不能用于SELECT语句.
  3. 用于处理选择列表项和输入宿主变量个数已经确定的动态SQL, 此种方法可以处理所有前两种方法能处理的情况, 此外, 还能处理SELECT语句.

由于第3种方法已经包含前两种方法的处理范围, 这篇文章我们主要介绍第3种方法.

1 介绍:

1.1 适用语句:

  • DML(Data Manipulation Language 数据操纵语言)
  • DDL(Data Definition Language 数据定义语言)
  • DCL(Data Control Language 数据控制语言)
  • 以及事务控制语句
  • 而且还可以处理SELECT语句 

   用SQL语句举例就是:

  • DELETE FROM tbl_name WHERE name=xxx
  • CREATE TABLE tbl_name(cola INT)
  • GRANT SELECT ON xx TO xxx
  • COMMIT
  • INSERT INTO tbl_name VALUES(:a)
  • DELETE FROM tbl_name WHERE name=:a
  • SELECT name FROM tbl_name WHERE no=xxx

1.2 不能处理的语句:

总体来讲就是对象名和列名不能使用宿主变量, 例如:

  • INSERT INTO :a VALUES(:b)
  • SELECT :a, :b FROM tbl_name WHERE name=:c

2 处理步骤:

2.1 PREPARE语句:

PREPARE命令用于明明和解析SQL语句, 语法如下:

EXEC SQL PREPARE  statement_name
    
FROM {:host_string | string_literal};

说明:

  • statement_name: 预编译器标识符.
  • host_string: 包含SQL语句的宿主变量.
  • string_literal: SQL语句文本字符串.

2.2 DECLARE游标:

使用PREPARE准备了SQL语句之后, 应该执行内嵌DECLARE命令定义游标, 语法如下:

EXEC SQL DECLARE cursor_name CURSOR
    
FOR statement_name;

说明:

  • cursor_name: 游标名.
  • statement_name: statement标识符, 在PREPARE过程中使用的.

2.3 OPEN打开游标:

当开始游标时, 会执行游标所对应的SQL语句, 语法如下:

EXEC SQL OPEN cursor_name [USING host_variable_list];

说明:

  • cursor_name: 游标名, 前面DECLARE的.
  • host_variable_list: 输入的宿主变量列表.

当执行OPEN命令时, 如果SQL语句不是SELECT语句, 会直接执行该语句, 此后可以关闭游标; 如果SQL语句是SELECT语句, 那么会将查询结果放到游标结果集中, 还必须使用内嵌的FETCH语句提取并处理游标结果集, 之后再关闭游标.

2.4 FETCH提取游标数据:

当SQL语句是SELECT语句时, OPEN之后会把结果存放到游标结果集中, 为了处理查询结果, 需要使用FETCH来提取数据, 语法如下:

EXEC SQL FETCH cursor_name INTO host_variable_list;

说明:

cursor_name: 游标名, 前面OPEN的.

host_variable_list: 输入的宿主变量列表.

2.5 CLOSE关闭游标:

提取并处理完游标数据后, 关闭游标, 语法如下:

EXEC SQL CLOSE cursor_name;

3 代码举例:

/* 大组成员表结构 */
struct  GrpMember
{
    
char str_bgID[22];  /* 大组ID */

    
char str_matID[22]; /* 素材ID */
    ...
    ...
};

/* *
 * 素材检索
 * 填冲好一个输入结构, 作为函数的参数进行调用.
 *
 * 这个输入结构中的各各成员项可以为空, 
 * 通过判断来确定数据库检索条件, 即: SELECT语句后面的条件.
 *
 
*/

void MatSearch(struct GrpMember *in )
{
    
/* 定义临时变量 */

    
char buf[256]; /* buffer */

    
/* 定义宿主变量 */
    EXEC SQL BEGIN DECLARE SECTION;
        
char sql_buf[1024]; /* 存放SQL语句 */
        
struct GrpMember ret; /* 查询结果 */
    EXEC SQL END DECLARE SECTION;

    
/*
     * 根据参数, 构造SQL语句
     *
     * 1 把SQL语句的前半部分拷贝到buffer中.
     * 2 判断输入参数的结构中各成员项是否为空, 如不为空, 则add到buffer中.
     
*/

    strcpy(sql_buf, 
"SELECT * FROM TBL_GRPMEMBER WHERE 1=1 " );
    
if (!
isEmpty(str_bgID))
    {
        sprintf(buf, 
"AND tbl_str_bgID = %s "
, str_bgID);
        strcat(sql_buf, buf);
    }
    ...
    ...

    
/* 准备动态SQL */

    EXEC SQL PREPARE sql_stmt FROM :sql_buf;

    
/* 定义游标 */
    EXEC SQL DECLARE c1 CURSOR FOR sql_stmt;

    
/* 打开游标: 执行查询 */
    EXEC SQL OPEN c1;

    
/* 提取查询结果 */
    
while(1 )
    {
        EXEC SQL FETCH c1 INTO ret; 
/* 此处伪代码, 应该逐项INTO, 逗号分割, 具体见游标的使用 */

    }

    EXEC SQL CLOSE c1;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值