前一天我列举了一些SQLite3库带的API,是SQLite的初级用法,今天我只讲一个API,但是用法会比前一次更好一点,便利一点。
还记得我们的sqlite3_exec函数么,今天就是说一下它的扩展用法。
函数原型
int sqlite3_exec(
sqlite3 *, /* An open database */
const char *sql, /* SQL to be evaluated */
int (*callback)(void*,int,char**,char**), /* 回调函数 */
void *, /* 传递给回调函数的参数 */
char **errmsg /* Error msg written here */
);
|
函数的作用就是执行sql语句,上次我们用这个函数执行的是CREATE TABLE命令,是不带返回值的。其实sqlite3_exec也能执行select语句,而且也能获得返回值,其中第三个参数(是个回调函数)起到了很大作用。
回调函数,就是在sqlite3_exec执行完成后自动被调用的函数。如果select返回多行数据,每返回一行回调函数将执行一次。(其中sqlite3_exec只用调用一次)
我来说明一下回调函数的各个参数。在sqlite3_exec的函数原型里可以看到,第三个参数是这么写的:
int (*callback)(void*,int,char**,char**)。
这就是回调函数的格式。回调函数中:
第一个参数是sqlite3_exec传进来的参数,也就是回调函数和外界交流的通道,比如我们可以传一个结构体指针地址进来,然后把查询到的数据保存在结构体中。
第二个参数是查询到的这一行总字段数n(总列数)。
第三个参数是一个数组,从0到n-1保存着咨询结果,都是字符串。
第四个参数和第三个对应,是每个字段的字段名,也是字符串。
知道了回调函数的格式,我们就可以利用sqlite3_exec来查询数据库并用回调函数处理查询结果了。函数可以这么写:
void AllSelect(sqlite3 * db) { char * errmsg; int rc = sqlite3_exec(db,"select * from addlist",GetDate,NULL,&errmsg); if(rc != SQLITE_OK) { printf("%sn",errmsg); } return ; } int GetDate(void * para,int n_column,char** data_column,char** name_column) { printf("编号:%d 姓名:%s 性别:%s 电话:%s Email:%s 地址:%sn", data_column[0],data_column[1],data_column[2],data_column[3], data_column[4],data_column[5]); return 0; }
相比于之前用sqlite3_prepare、sqlite3_step要简单很多吧?不过它也有局限性,如果数据库里保存着的是二进制文件(图片、压缩包什么的),就不能用了这个方法了。
还有,前一次没有注意到的,sqlite3_exec的最后一个参数errmsg,以前都传入的是NULL,其实对于数据库来说,这个参数是很有作用的,它可以获取数据库出错信息。正如我这次写的代码一样,如果sqlite3_exec执行不成功,则printf("%sn",errmsg);
当初选择ngx_lua_waf作为自己的WAF,主要原因就是因为其可扩展性与性能上有一个很好的平衡。 lua语言的灵活性与效率是很多脚本层WAF无可匹及的。 ngx_lua_waf自身是比较简单的,而且存在很多误报、漏报、绕过的现象,我整理如下,来改进自己的w…
请登录后发表评论
注册