【SQL】宿主语言接口
一般情况下,SQL语句是嵌套在宿主语言(如C语言)中的。有两种嵌套方式:
1.调用层接口(CLI):提供一些库,库中的函数和方法实现SQL的调用
2.直接嵌套SQL:在代码中嵌套SQL语句,提交给预处理器,将SQL语句转换成对宿主语言有意义的内容,如调用库中的函数和方法代替SQL语句
阻抗不匹配问题:连接SQL语句与常规编程语言的基本问题。SQL核心使用关系数据模型,而C语言等常规语言则使用整型、实数型、算数型、字符型、指针等类型。
一、SQL与宿主语言连接
EXEC SQL关键字提示预处理将由SQL代码进入。
共享变量:实现数据库和宿主语言之间的信息交换。
SQL中共享变量要加冒号,宿主语言中不需要。
SQLSTATE:特殊变量,由5个字符的数组组成(在C中分配空间要分配6个,后面有'\0'),存放表示调用过程出现的问题的代码。'00000'表示没有错误。
1.1 DECLARE节
声明共享变量要在声明节中。
EXEC SQL BEGIN DECLARE SECTION;
...
EXEC SQL END DECLARE SECTION;
EXEC SQL BEGIN DECLARE SECTION;
char studioName[], studioAddr[];
char SQLSTATE[];
EXEC SQL END DECLARE SECTION;
1.2 使用共享变量
任何不含返回结果的SQL语句,都可以用EXEC SQL为前缀嵌入宿主语言。
void getStudio(){
EXEC SQL BEGIN DECLARE SECTION;
char studioName[], studioAddr[];
char SQLSTATE[];
EXEC SQL END DECLARE SECTION;
printf("please input studio name:\n");
scanf("%s", &studioName);
printf("please input studio address:\n");
scanf("%s", &studioAddr);
EXEC SQL INSERT INTO Studio(name, address)
VALUES (:studioName, :studioAddr);
1.3 连接SQL返回结果到宿主语言
因为阻抗不匹配,使得返回结果无法直接返回到宿主语言中,必须使用下面两种机制中的一个。
单元组选择语句:只有一个结果的查询语句,将该元组存储到共享变量中。
游标:为查询声明一个游标,游标范围覆盖结果关系中的所有元组,每个元组一次被提取到共享变量,由宿主语言处理。
1.4单元组选择语句
类似于普通的select-from-where语句,只是SELECT子句后面紧跟着关键字INTO和一连串的共享变量。如果结果少于或多于一个元组,则共享变量中不会有值,且SQLSTATE中存入一个错误码。
void printNetWorth(){
EXEC SQL BEGIN DECLARE SECTION;
char studioName[];
int presNetWorth;
char SQLSTATE[];
EXEC SQL END DECLARE SECTION;
/*输入studioName的代码*/
EXEC SQL SELECT netWorth
INTO :presNetWorth
FROM Studio, MovieExec
WHERE presC# = cert# AND
Studio.name = :studioName;
/*检查SQLSTATE中的代码是否为'00000'*/
}
1.5 游标
游标声明:
EXEC SQL DECLARE 游标名称 CURSOR FOR 查询
初始化游标的位置:使之指向第一个元组
EXEC SQL OPEN 游标名称
得到下一个元组:
EXEC SQL FETCH FROM 游标名称 INTO 变量列表
变量列表中存放获取的结果,如果已经遍历结束则SQLSTATE中返回'02000'
关闭游标:
EXEC SQL CLOSE 游标名称
void worthRanges(){
int i, digits, counts[];
EXEC SQL BEGIN DECLARE SECTION;
int worth;
char SQLSTATE[];
EXEC SQL END DECLARE SECTION;
EXEC SQL DECLARE execCursor CURSOR FOR
SELECT netWorth FROM MovieExec;
EXEC SQL OPEN execCursor;
for(i = 1; i < 15; i++)
counts[i] = 0;
while(1){
EXEC SQL FETCH FROM execCursor INTO :worth;
if(NO_MORE_TUPLES)
break;
digits = 1;
while((worth /= 10) > 0) digits++;
if(digits <= 14) counts[digits]++;
}
EXEC SQL CLOSE execCursor;
for(i = 1; i < 15; i++)
printf("digits = %d: number of execs = %d\n",i, counts[i]);
}
1.6 游标更新
可以通过游标删除或修改当前的元组。
WHERE后只能是 WHERE CURRENT OF
下面的例子将资产少于1000的删除,多余1000的翻倍。
#define NO_MORE_TUPLES !(strcmp(SQLSTATE,"02000"))
void changeWorth(){
EXEC SQL BEGIN DECLARE SECTION;
int certNo, worth;
char execName[], execAddr[], SQLSTATE[];
EXEC SQL END DECLARE SECTION;
EXEC SQL DECLARE execCursor CURSOR FOR MovieExec;
EXEC SQL OPEN execCursor;
while(1){
EXEC SQL FETCH FROM execCursor INTO :execName, :execAddr, :certNo, :worth;
if(NO_MORE_TUPLES) break;
if(worth < 1000)
EXEC SQL DELETE FROM MovieExec
WHERE CURRENT OF execCursor;
else
EXEC SQL UPDATE MovieExec
SET netWorth = 2 * netWorth
WHERE CURRENT OF execCursor;
}
EXEC SQL CLOSE execCursor;
}
1.7 避免并发修改
如果在游标获取到元组到游标关闭前,不希望元组发生变化:
EXEC SQL DECLARE 游标名 INSENSITIVE CURSOR FOR 查询;
不希望通过游标修改元组
EXEC SQL DECLARE 游标名 CURSOR FOR 查询 FOR READ ONLY;
1.8动态SQL
在运行时,通过输入SQL语句实现查询需要动态SQL语句支持。
EXEC SQL PREPARE V FROM 表达式;
EXEC SQL EXECUTE V; //这句可以多次执行 表明表达式V执行了多次
上面两句合并后可以写成:
EXEC SQL EXECUTE IMMEDIATE 表达式;
void readQuery(){
EXEC SQL BEGIN DECLARE SECTION;
char * query;
EXEC SQL END DECLARE SECTION;
/*输入SQL语句,放入query中*/
EXEC SQL PREPARE SQLquery FROM :query;
EXEC SQL EXECUTE SQLquery;
}
【SQL】宿主语言接口的更多相关文章
- [转]SQLITE3 C语言接口 API 函数简介
SQLITE3 C语言接口 API 函数简介 说明:本说明文档属作者从接触 SQLite 开始认识的 API 函数的使用方法, 由本人翻译, 不断更新. /* 2012-05-25 */ int sq ...
- 基于Oracle OCI的数据访问C语言接口ORADBI .
基于Oracle OCI的数据访问C语言接口ORADBI cheungmine@gmail.com Mar. 22, 2008 ORADBI是我在Oracle OCI(Oracle 调用接口)基础 ...
- Swift中对C语言接口缓存的使用以及数组、字符串转为指针类型的方法
由于Swift编程语言属于上层编程语言,而Swift中由于为了低层的高性能计算接口,所以往往需要C语言中的指针类型,由此,在Swift编程语言刚诞生的时候就有了UnsafePointer与Unsafe ...
- C语言接口与实现实例
一个模块有两部分组成:接口和实现.接口指明模块要做什么,它声明了使用该模块的代码可用的标识符.类型和例程,实现指明模块是如何完成其接口声明的目标的,一个给定的模块通常只有一个接口,但是可能会有许多种实 ...
- SQL设置语言,返回中文”星期几”格式
SQL中语言表: SELECT * FROM sys.syslanguages eg: SET LANGUAGE 简体中文 --设置语言 PRINT DATENAME(weekday,GETDAT ...
- SQL Server数据库(SQL Sever语言 CRUD)
使用SQL Sever语言进行数据库的操作 常用关键字identity 自增长primary key 主键unique 唯一键not null 非空references 外键(引用) 在使用查询操作数 ...
- 18 A GIF decoder: an exercise in Go interfaces 一个GIF解码器:go语言接口训练
A GIF decoder: an exercise in Go interfaces 一个GIF解码器:go语言接口训练 25 May 2011 Introduction At the Googl ...
- SQLAlchemy 学习笔记(一):Engine 与 SQL 表达式语言
个人笔记,如有错误烦请指正. SQLAlchemy 是一个用 Python 实现的 ORM (Object Relational Mapping)框架,它由多个组件构成,这些组件可以单独使用,也能独立 ...
- opencv的C语言接口和C++接口差别(入门篇)
opencv是一个开源的图像处理库,最经典的1.0版本号提供的接口都是C语言接口. 后来的opencv2.x版本号保留了C语言接口,可是提供了C++接口,当中的C语言接口仅仅是为了向后兼容,而C++接 ...
随机推荐
- Hadoop执行bin/stop-all.sh时no namenode to stop异常
1 先关闭hadoop所有服务 命令: bin/stop-all.sh 2 格式化namenode 命令: bin/hadoop namenode -format 3 重新启动所有服务 命令: bin ...
- [Elasticsearch] 多字段搜索 (五) - 以字段为中心的查询
以字段为中心的查询(Field-centric Queries) 上述提到的三个问题都来源于most_fields是以字段为中心(Field-centric),而不是以词条为中心(Term-centr ...
- 一种保持顺序的Properties
其实properties有没有顺序都一样 程序都能正常运行 但看着就比较闹心 所以网上找了找 还真有人给了个例子实现读Property的有序 但是删除某些属性之后 写入又有问题 会异常 后来重写了一下 ...
- wait_event_interruptible_timeout
最近一套方案涉及到内核线程之间的同步,用到了函数wait_event_interruptible_timeout函数,大致是这样: A:是一个后台的线程,平常没事就睡觉,有时被唤醒,或者每5分钟醒一次 ...
- 【bzoj2809】[Apio2012]dispatching 贪心+可并堆
题目描述 在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿.在这个帮派里,有一名忍者被称之为 Master.除了 Master以外,每名忍者都有且仅有一个上级.为保密,同时增 ...
- Android中有哪些好的开发框架?
在安卓开发中,框架的使用必不可少,合理利用一些好的开发框架,往往可以达到事半功倍的效果.本文小编就将和大家分享安卓开发者不得不知的5款框架,一起来看看吧,新技能get走起~~ 1.thinkAndro ...
- JS设置cookie,读取cookie,删除cookie
总结了一下cookie的使用,不全面.都是基础的知识,后期还会再添加. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitiona ...
- TCP/IP Note3
TCP/IP协议 TCP/IP是不同的通信协议的大集合. 协议族:TCP/IP是基于TCP和IP这两个最初的协议之上的不同的通信协议的大集合. 1. TCP - 传输控制协议 TCP用于从应用程序到网 ...
- BZOJ1607 [Usaco2008 Dec]Patting Heads 轻拍牛头 【筛法】
题目 今天是贝茜的生日,为了庆祝自己的生日,贝茜邀你来玩一个游戏. 贝茜让N(1≤N≤100000)头奶牛坐成一个圈.除了1号与N号奶牛外,i号奶牛与i-l号和i+l号奶牛相邻.N号奶牛与1号奶牛相邻 ...
- Splunk学习与实践
一. Splunk公司与产品 美国Splunk公司,成立于2004年,2012年纳斯达克上市,第一家大数据上市公司,荣获众多奖项和殊荣.总部位于美国旧金山,伦敦为国际总部,香港设有亚太支持中心, ...