MFC中sqlite3的使用

  • sqlite3介绍
  • sqlite3安装
  • 常用API函数
    • 操作流程
    • 接口函数
      • 执行sql语句函数
      • 回调函数
  • MFC中案例实践
  • 控制台实践

sqlite3介绍

SQLite 是一个软件库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是在世界上最广泛部署的 SQL 数据库引擎。SQLite引擎不是个程序与之通信的独立进程,而是连接到程序中成为它的一个主要部分。所以主要的通信协议是在编程语言内的直接API调用。这在消耗总量、延迟时间和整体简单性上有积极的作用。整个数据库(定义、表、索引和数据本身)都在宿主主机上存储在一个单一的文件中。它的简单的设计是通过在开始一个事务的时候锁定整个数据文件而完成的。

sqlite3安装

查看我的另一篇文章:传送链接:

常用API函数

sqlite提供的是一些C函数接口,你可以用这些函数操作数据库。通过使用这些接口,传递一些标准 sql 语句(以 char * 类型)给 sqlite 函数,sqlite 就会为你操作数据库。

sqlite 跟微软的access一样是文件型数据库,就是说,一个数据库就是一个文件,此数据库里可以建立很多的表,可以建立索引、触发器等等,但是,它实际上得到的就是一个文件。备份这个文件就备份了整个数据库。

sqlite 不需要任何数据库引擎,这意味着如果你需要 sqlite 来保存一些用户数据,不需要安装数据库,只需要对你需要使用的数据库进行拷贝就行了。

数据库定义、操作、查询等操作;
DDL - 数据定义语言
CREATE 创建一个新的表,一个表的视图,或者数据库中的其他对象。
ALTER 修改数据库中的某个已有的数据库对象,比如一个表。
DROP 删除整个表,或者表的视图,或者数据库中的其他对象。
DML - 数据操作语言

INSERT 创建一条记录。
UPDATE 修改记录。
DELETE 删除记录。
DQL - 数据查询语言

SELECT 从一个或多个表中检索某些记录。

操作流程

  1. 定义 sqlit3 *数据

sqlite 里最常用到的是 sqlite3 * 类型。从数据库打开开始,sqlite就要为这个类型准备好内存,直到数据库关闭,整个过程都需要用到这个类型。当数据库打开时开始,这个类型的变量就代表了你要操作的数据库。

  1. 打开数据库

int sqlite3_open( 数据库文件名, sqlite3 ** );
用这个函数开始数据库操作。需要传入两个参数,一是数据库文件名,比如:Database.db。
文件名不需要一定存在,如果此文件不存在,sqlite 会自动建立它。如果它存在,就尝试把它当数据库文件来打开。
sqlite3 ** 参数即前面提到的关键数据结构。这个结构底层细节如何,你不要关它。

函数返回值表示操作是否正确,如果是 SQLITE_OK 则表示操作正常。相关的返回值sqlite定义了一些宏。具体这些宏的含义可以参考 sqlite3.h 文件。里面有详细定义(顺便说一下,sqlite3 的代码注释率自称是非常高的,实际上也的确很高。只要你会看英文,sqlite 可以让你学到不少东西)。

  1. 关闭数据库
    int sqlite3_close(sqlite3 *);
    这个函数经常和sqlite3_open成对使用,当前面打开一个数据库时这里一定要记得关闭;
    整个过程演示:
 #include "sqlite3.h"
int main( int , char** )
{
sqlite3 * db = NULL; //声明sqlite关键结构指针int result;
//打开数据库
//需要传入 db 这个指针的指针,因为 sqlite3_open 函数要为这个指针分配内存,还要让db指针指向这个内存区
result = sqlite3_open( “database.db”, &db );
if( result != SQLITE_OK )
{
//数据库打开失败
return -1;
}
//数据库操作代码
//…
//数据库打开成功
//关闭数据库
sqlite3_close( db );
return 0;
}

接口函数

执行sql语句函数

int sqlite3_exec(sqlite3*, const char *sql, sqlite3_callback, void *, char **errmsg )

这就是执行一条 sql 语句的函数。
第1个参数是前面open函数得到的指针。说了是关键数据结构。
第2个参数const char sql 是一条 sql 语句,以\0结尾。
第3个参数sqlite3_callback 是回调,当这条语句执行之后,sqlite3会去调用你提供的这个函数。
第4个参数void * 是你所提供的指针,你可以传递任何一个指针参数到这里,这个参数最终会传到回调函数里面,如果不需要传递指针给回调函数,可以填NULL。
第5个参数char ** errmsg 是错误信息。注意是指针的指针。
sqlite3里面有很多固定的错误信息。执行 sqlite3_exec 之后,执行失败时可以查阅这个指针(直接 printf(“%s\n”,errmsg))得到一串字符串信息,这串信息告诉你错在什么地方。sqlite3_exec函数通过修改你传入的指针的指针,把你提供的指针指向错误提示信息,这样sqlite3_exec函数外面就可以通过这个 char
得到具体错误提示。

说明:通常,sqlite3_callback 和它后面的 void * 这两个位置都可以填 NULL。填NULL表示你不需要回调。比如你做 insert 操作,做 delete 操作,就没有必要使用回调。

而当做 select 时,就要使用回调,因为 sqlite3 把数据查出来,得通过回调告诉你查出了什么数据。

回调函数

exec 的回调函数

typedef int (sqlite3_callback)(void,int,char**, char**);
回调函数必须定义成上面这个函数的类型,每查到符合sql语句的结果就执行一次回调函数。
回调函数必须定义成上面这个函数的类型,每查到符合sql语句的结果就执行一次回调函数。
回调函数必须定义成上面这个函数的类型,每查到符合sql语句的结果就执行一次回调函数。
回调函数必须定义成上面这个函数的类型,每查到符合sql语句的结果就执行一次回调函数。

//sqlite3的回调函数 //sqlite 每查到一条记录,就调用一次这个回调 int LoadMyInfo(void* para,intn_column,char** column_value,char** column_name);
//para是你在sqlite3_exec 里传入的void参数通过para参数,你可以传入一些特殊的指针(比如类指针、结构指针),
//然后在这里面强制转换成对应的类型(这里面是void
类型,必须强制转换成你的类型才可用)。
//n_column是这一条记录有多少个字段(即这条记录有多少列)
//char** column_value 是个关键值,查出来的数据都保存在这里,它实际上是个1维数组(不要以为是2维数组),
//每一个元素都是一个char*值,是一个字段内容(用字符串来表示,以\0结尾)

//char** column_name 跟column_value是对应的,表示这个字段的字段名称

下面给个简单的例子:

//sqlite3的回调函数
// sqlite 每查到一条记录,就调用一次这个回调
int LoadMyInfo( void * para, int n_column, char ** column_value, char ** column_name )
{
//para是你在 sqlite3_exec 里传入的 void * 参数
//通过para参数,你可以传入一些特殊的指针(比如类指针、结构指针),然后在这里面强制转换成对应的类型(这里面是void*类型,必须强制转换成你的类型才可用)。然后操作这些数据
//n_column是这一条记录有多少个字段 (即这条记录有多少列)
// char ** column_value 是个关键值,查出来的数据都保存在这里,它实际上是个1维数组(不要以为是2维数组),每一个元素都是一个 char * 值,是一个字段内容(用字符串来表示,以\0结尾)
//char ** column_name 跟 column_value是对应的,表示这个字段的字段名称
//这里,我不使用 para 参数。忽略它的存在.
int i;
printf( “记录包含 %d 个字段\n”, n_column );
for( i = 0 ; i < n_column; i ++ )
{
printf( “字段名:%s ß> 字段值:%s\n”, column_name[i], column_value[i] );
}
printf(------------------\n“ );
return 0;
}int main( int , char ** )
{
sqlite3 * db;
int result;
char * errmsg = NULL;
result = sqlite3_open( “c:\\Dcg_database.db”, &db );
if( result != SQLITE_OK )
{
//数据库打开失败
return -1;
}//数据库操作代码
//创建一个测试表,表名叫 MyTable_1,有2个字段: ID 和 name。其中ID是一个自动增加的类型,以后insert时可以不去指定这个字段,它会自己从0开始增加result = sqlite3_exec( db, “create table MyTable_1( ID integer primary key autoincrement, name nvarchar(32) ), NULL, NULL, errmsg );
if(result != SQLITE_OK )
{
printf( “创建表失败,错误码:%d,错误原因:%s\n”, result, errmsg );
}
//插入一些记录
result = sqlite3_exec( db, “insert into MyTable_1( name ) values ( ‘走路’ ), 0, 0, errmsg );
if(result != SQLITE_OK )
{
printf( “插入记录失败,错误码:%d,错误原因:%s\n”, result, errmsg );
}
result = sqlite3_exec( db, “insert into MyTable_1( name ) values ( ‘骑单车’ ), 0, 0, errmsg );
if(result != SQLITE_OK )
{
printf( “插入记录失败,错误码:%d,错误原因:%s\n”, result, errmsg );
}
result = sqlite3_exec( db, “insert into MyTable_1( name ) values ( ‘坐汽车’ ), 0, 0, errmsg );
if(result != SQLITE_OK )
{
printf( “插入记录失败,错误码:%d,错误原因:%s\n”, result, errmsg );
}
//开始查询数据库
result = sqlite3_exec( db, “select * from MyTable_1”, LoadMyInfo, NULL, errmsg );
//关闭数据库
sqlite3_close( db );
return 0;
}

通过上面的例子,应该可以知道如何打开一个数据库,如何做数据库基本操作。

有这些知识,基本上可以应付很多数据库操作了。

不适用回调函数查询数据库参考链接:https://www.cnblogs.com/ransn/p/8063962.html

MFC中案例实践

MFC案例:
在这里插入图片描述


//将wchar_t* 转成char*的实现函数如下:char *w2c(char *pcstr, wchar_t *pwstr, size_t len){int nlength = wcslen(pwstr);//获取转换后的长度int nbytes = WideCharToMultiByte(0, // specify the code page used to perform the conversion0,         // no special flags to handle unmapped characterspwstr,     // wide character string to convertnlength,   // the number of wide characters in that stringNULL,      // no output buffer given, we just want to know how long it needs to be0,NULL,      // no replacement character givenNULL);    // we don't want to know if a character didn't make it through the translation// make sure the buffer is big enough for this, making it larger if necessaryif (nbytes > len)   nbytes = len;// 通过以上得到的结果,转换unicode 字符为ascii 字符WideCharToMultiByte(0, // specify the code page used to perform the conversion0,         // no special flags to handle unmapped characterspwstr,   // wide character string to convertnlength,   // the number of wide characters in that stringpcstr, // put the output ascii characters at the end of the buffernbytes,                           // there is at least this much space thereNULL,      // no replacement character givenNULL);return pcstr;}int callback(void* para, int ncount, char** col_value, char** col_name)
{TRACE("total column is %d\n", ncount);TRACE("start>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>", ncount);for (int i = 0; i < ncount; i++){password = col_value[0];select_name = col_name[0];TRACE("col_name:%s----> clo_value:%s\n", col_name[i], col_value[i]);}return 0;
}bool UserPermission::UserInformationSelect()
{// TODO: 在此处添加实现代码.sqlite3 * pDB = NULL;char* cErrMsg;int ret = sqlite3_open("test.db", &pDB);if (ret == SQLITE_OK){MessageBox(TEXT("test.db打开成功!"));}else{MessageBox(TEXT("test.db打开失败!"));return false;}CString sqlcommand;CString test,current_username;GetDlgItem(IDC_EDIT1)->GetWindowText(test);int user_index;user_index = m_ctrlUserName.GetCurSel();switch (user_index){case 0:{current_username = "manager";break;}case 1:{current_username = "operater";break;}case 3:{current_username = "server";break;}default:break;}/*string strSql;strSql = "select * from userinfo";int res = sqlite3_exec(pDB, strSql.c_str(), callback, NULL, &cErrMsg);*/char sqlcommand1[1280];memset(sqlcommand1, 0, sizeof(sqlcommand1));sqlcommand.Format(_T("SELECT * FROM user where id = 1")) ;//strcpy(sqlcommand1, sqlcommand.GetBuffer(sqlcommand.GetLength()));w2c(sqlcommand1, sqlcommand.GetBuffer(), 1280);int res = sqlite3_exec(pDB, sqlcommand1, callback, NULL, &cErrMsg);if (res != SQLITE_OK){CString msg;msg.Format(_T("ret=%d\n sql=[%s]\n"), ret, sqlcommand);MessageBox(msg);MessageBox(TEXT("select fail:"));return false;}//delete[] sqlcommand1;sqlite3_close(pDB);return true;
}

控制台实践

#include <iostream>
using namespace std;
#include "sqlite/sqlite3.h"
int callback(void*,int,char**,char**);
int main()
{sqlite3* db;int nResult = sqlite3_open("test.db",&db);if (nResult != SQLITE_OK){cout<<"打开数据库失败:"<<sqlite3_errmsg(db)<<endl;return 0;}else{cout<<"数据库打开成功"<<endl;}char* errmsg;nResult = sqlite3_exec(db,"create table fuck(id integer primary key autoincrement,name varchar(100))",NULL,NULL,&errmsg);if (nResult != SQLITE_OK){sqlite3_close(db);cout<<errmsg;sqlite3_free(errmsg);return 0;}string strSql;strSql+="begin;\n";for (int i=0;i<100;i++){strSql+="insert into fuck values(null,'heh');\n";}strSql+="commit;";//cout<<strSql<<endl;nResult = sqlite3_exec(db,strSql.c_str(),NULL,NULL,&errmsg);if (nResult != SQLITE_OK){sqlite3_close(db);cout<<errmsg<<endl;sqlite3_free(errmsg);return 0;}strSql = "select * from fuck";nResult = sqlite3_exec(db,strSql.c_str(),callback,NULL,&errmsg);if (nResult != SQLITE_OK){sqlite3_close(db);cout<<errmsg<<endl;sqlite3_free(errmsg);return 0;}sqlite3_close(db);return 0;
}int callback(void* ,int nCount,char** pValue,char** pName)
{string s;for(int i=0;i<nCount;i++){s+=pName[i];s+=":";s+=pValue[i];s+="\n";}cout<<s<<endl;return 0;
}

链接:

MFC ListControl与SQLite系列
MFC ListControl与SQLite(一)SQLite的使用
MFC ListControl与SQLite(二)用C++读SQLite
MFC ListControl与SQLite(三)用C++增删改查SQLite表格数据
MFC ListControl与SQLite(四)List Control与SQLite结合实例

原文链接:https://blog.csdn.net/shanbadizun/article/details/113535468

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/94149.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/94149.shtml
英文地址,请注明出处:http://en.pswp.cn/web/94149.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LeetCode第1019题 - 链表中的下一个更大节点

题目 解答 class Solution {Stack<Integer> stack new Stack<>();List<Integer> values new LinkedList<>();public int[] nextLargerNodes(ListNode head) {nextLargerNodes2(head);return values.stream().mapToInt(x -> x).toArray();}publi…

STM32 硬件I2C读写MPU6050

本文代码基于 STM32 单片机&#xff0c;通过 I2C 总线驱动 MPU6050 六轴传感器&#xff08;集成加速度计与陀螺仪&#xff09;&#xff0c;实现传感器初始化、ID 读取、原始数据采集&#xff0c;并借助 OLED 显示屏实时展示加速度&#xff08;AccX、AccY、AccZ&#xff09;与角…

倍福下的EC-A10020-P2-24电机调试说明

今天调试EC-A10020-P2-24电机&#xff0c;采用力位混合控制指令进行控制&#xff0c;无前馈力矩&#xff0c;只调节Kp和Kd,跟踪红色轨迹&#xff08;正弦信号&#xff1a;幅值10&#xff0c;频率0.5Hz&#xff09;&#xff0c;结果显示Kp 180, Kd 40&#xff0c;实际上Kp进一步…

SQL注入1----(sql注入原理)

一.前言前面我们讲解了一下信息收集&#xff0c;本章节我们来讲解一下sql注入的基本原理&#xff0c;我们拿之前搭建的测试网站pikachu来测试&#xff0c;对应工具包也已经放在了工具里面&#xff0c;大家可以自行去下载。SQL注入攻击漏洞的原因&#xff0c;是由于程序员在编写…

C++智能指针详解:用法与实践指南

C智能指针详解&#xff1a;用法与实践指南 在C编程中&#xff0c;动态内存管理始终是开发者面临的重要挑战。手动分配和释放内存不仅繁琐&#xff0c;还容易因疏忽导致内存泄漏、悬垂指针等问题。为解决这些痛点&#xff0c;C标准库引入了智能指针&#xff08;Smart Pointers&a…

fastdds qos:DurabilityQosPolicy

假如DataWriter先起来&#xff0c;并且已经写了一些数据&#xff0c;之后有新的DataReader起来&#xff0c;那么新起来的DataReader能不能接收到它启动之前&#xff0c;DataWriter发布的数据呢。DurabilityQosPolicy用来做这种控制。VOLATILE_DURABILITY_QOS&#xff1a;易失的…

【读代码】SQLBot:开源自然语言转SQL智能助手原理与实践

一、项目简介 SQLBot 是 DataEase 团队开源的自然语言转 SQL 智能助手,致力于让非技术用户也能通过自然语言与数据库对话,自动生成 SQL 查询,实现自助数据分析、智能BI问答、报表生成等场景。SQLBot 结合了大语言模型(LLM)、数据库元数据解析、SQL解析与执行等多项技术,…

开题报告被退回?用《基于大数据的慢性肾病数据可视化分析系统》的Hadoop技术,一次通过不是梦

&#x1f496;&#x1f496;作者&#xff1a;计算机编程小咖 &#x1f499;&#x1f499;个人简介&#xff1a;曾长期从事计算机专业培训教学&#xff0c;本人也热爱上课教学&#xff0c;语言擅长Java、微信小程序、Python、Golang、安卓Android等&#xff0c;开发项目包括大数…

HEVC(H.265)与HVC1的关系及区别

HEVC&#xff08;H.265&#xff09;与HVC1的关系及区别可归纳如下&#xff1a;一、技术定义差异‌HEVC&#xff08;H.265&#xff09;‌国际标准化组织制定的通用视频编码标准&#xff0c;由ITU-T和ISO/IEC联合开发‌1支持8K分辨率&#xff0c;压缩效率较H.264提升约50%‌1‌HV…

Java获取被nginx代理的emqx客户端真实ip

Java获取被nginx代理的emqx客户端真实ip 契机 ⚙ 使用nginx作为负载均衡&#xff08;Load Balancing&#xff09;的时候&#xff0c;发现真实ip无法获取。几经折腾终于拿到真实ip&#xff0c;又发现被代理的端口又无法使用非代理模式连接&#xff0c;由于之前暴露的docker端口有…

Jenkins自动化部署服务到Kubernetes环境

在现代软件开发中,持续集成和持续部署(CI/CD)已成为提高开发效率和软件质量的关键实践。本文将介绍如何使用Jenkins自动化部署服务到Kubernetes环境,并重点介绍Maven与私服的配置。 环境准备 在开始之前,请确保您已准备好以下环境: Jenkins服务器 Kubernetes集群 Docker镜…

OpenAI重新开源!gpt-oss-20b适配昇腾并上线魔乐社区

2025年8月5日&#xff0c;OpenAI发布了两款全新的开源权重语言模型&#xff0c;均为混合专家&#xff08;MoE&#xff09;架构&#xff0c;其规模设计可在消费级GPU和云端的多种硬件上高效运行。这些模型采用 Apache 2.0 许可协议&#xff0c;因此可用于蒸馏到其他推理模型中、…

SpringCloud入门(简洁明了)

目录 一.创建微服务项目 (一)环境准备 (二)项目结构图 (三)流程 二. Nacos (一)注册中心 1.服务注册 2.服务发现 3.编写微服务API 4.远程调用基本实现 5.负载均衡 6.LoadBalanced注解式注解均衡 7.注册中心宕机&#xff0c;远程调用还能成功吗 (二)配置中心 1.基…

集成算法学习笔记

一、集成算法简介1. 核心思想类比“多个专家综合判断优于单个专家”&#xff0c;通过构建并结合多个个体学习器&#xff0c;提升模型的泛化能力&#xff08;降低过拟合风险、提高预测准确性&#xff09;&#xff0c;完成复杂的学习任务。2. 个体学习器与结合模块个体学习器&…

让Chrome信任自签名证书

让Chrome信任自签名证书&#xff08;Unix系列OS&#xff09; 背景 想在本地测试自己写的基于HTTPS连接的Web应用&#xff0c;跑在3001端口。但使用Chrome浏览器访问https://localhost:3001时显示连接不安全。解决了但没解决 使用mkcert一键创建证书&#xff1a; mkcert localho…

[江科大库]基于 OpenMV 的矩形识别与 STM32 串口通信(电子设计大赛实用教程)

🎯 基于 OpenMV 的矩形识别与 STM32 串口通信(电子设计大赛实用教程) 一、前言 在本科生电子设计大赛中,经常会遇到图像识别相关的任务,例如: 识别 矩形框(如识别一个 A4 纸、黑色标记框等); 将识别结果传输到 STM32 单片机,用于后续控制(舵机、移动小车、机械臂…

人脸识别驱动的工厂人体属性检测与预警机制

人体属性检测&#xff1a;人脸识别智慧检测驱动的工厂管理革新&#xff08;所有图片均为真实项目案例&#xff09;在制造业数字化转型浪潮中&#xff0c;人体属性检测技术已成为破解传统工厂管理难题的核心工具。通过融合人脸识别智慧检测、目标检测算法与多模态数据分析&#…

数据工程师——ETL

ETL面试题01 一、基础概念与理论类 1. 请解释什么是 ETL?它在数据处理流程中扮演什么角色? 答:ETL就是数据抽取、转化、加载。目的是将分散的数据源集中在一起进行处理分析。 数据抽取:是指各种数据源中抽取数据,包括关系型数据库(MySQL、Oracle等)、日志文件、Exce…

Oracle APEX 经典报表中的Checkbox

目录 1. 建表&#xff06;投入测试数据 2. 经典报表做成 2-1. 画面布局如下​编辑 2-2. 报表使用的SQL 2-3. RS列的Heading设定 2-4. Function and Global Variable Declaration 2-5. Execute when Page Loads 2-6. Process 3. 运行效果​编辑 1. 建表&#xff06;投入…

Codeforces Round 1043 (Div.3)

比赛连接&#xff1a;Codeforces Round 1043 (Div.3) A. Homework 题目链接&#xff1a;A - Homework Vlad and Dima have been assigned a task in school for their English class. They were given two strings aaa and bbb and asked to append all characters from bbb …