8、Linux C/C++ 实现MySQL的图片插入以及图片的读取
本文结合了Linux C/C++ 实现MySQL的图片插入以及图片的读取,特别是数据库读写的具体流程
一、文件读取相关函数
- fseek() 可以将文件指针移动到文件中的任意位置。其基本形式如下:
int fseek(FILE *stream, long offset, int whence);
其中,stream 是一个指向已经打开的文件流的指针;offset 是相对于 whence 参数所表示位置的偏移量,可以为正数、负数或者零;而 whence 则是指定了 offset 所表示偏移量相对位置的标志。具体来说,它可以取以下三个值之一:
– SEEK_SET: 表示从文件起始处开始计算偏移量;
– SEEK_CUR: 表示从当前位置开始计算偏移量;
– SEEK_END: 表示从文件结尾处开始计算偏移量。
当调用成功时,返回值为 0;否则返回非零值。
需要注意的是,在使用完 fseek() 函数后,我们需要通过调用 ftell() 函数获取当前文件指针所在位置,并且确保该位置与我们想要设置的位置一致。
- ftell() 用于获取文件指针当前位置的函数,其原型如下:
long ftell(FILE *stream);
其中,stream 表示要获取当前位置的文件流指针。调用成功后,返回值为 long 类型,表示当前文件指针相对于文件开头的偏移量(以字节为单位)。
- fread() 用于从文件中读取二进制数据并存储到内存缓冲区中,其原型如下:
size_t fread(void *ptr, size_t size, size_t count, FILE *stream);
其中,ptr: 读取后存放数据的缓冲区地址。size: 每次读取元素的字节数。count: 读取元素的个数。stream: 文件指针。调用成功时,返回值为实际读取到的元素数量,通常与 count 参数相等。如果在读取操作过程中遇到了文件结束符或者发生了错误,则返回值会小于 count。
- fwrite()用于将内存中的二进制数据按照指定数量和大小写入到文件中。其原型如下:
size_t fwrite(void *ptr, size_t size, size_t count, FILE *stream);
其中,ptr:指向要写入的数据的指针。size:每个数据块的大小(以字节为单位)。count:要写入的数据块数量。stream:指向目标文件的FILE结构体指针。
二、stmt相关函数
使用stmt(或者叫做statement)能够将SQL语句预编译,这可以提高代码的执行效率和安全性。当我们使用stmt时,可以将 SQL 查询中需要传递的参数占位符(例如 “?”)和实际的参数值进行绑定,随后数据库会先对SQL语句进行预编译,并生成一个执行计划。之后,每次执行该SQL语句时,只需将实际参数值填充到占位符位置上即可执行该计划,而不需要重新编译整个SQL语句。这样可以大大减少数据库系统的开销,并提高应用程序的响应速度。
1、将本地(图片)数据写入到数据库的具体流程如下
1、创建一个MYSQL_STMT *stmt用于表示预处理语句
2、使用mysql_stmt_init()函数初始化MYSQL_STMT结构体对象stmt
3、使用mysql_stmt_prepare()函数将要执行的SQL语句绑定到对象stmt上
4、创建一个结构体MYSQL_BIND *param用于绑定SQL语句中参数的变量
5、使用mysql_stmt_bind_param()函数设置SQL语句中所需要的参数
6、使用mysql_stmt_send_long_data()函数绑定二进制数据到预处理的SQL语句的占位符中
7、使用mysql_stmt_execute()函数执行SQL语句
8、使用mysql_stmt_fetch()函数获取结果集中的数据
9、在使用完毕后,使用mysql_stmt_close()函数释放MYSQL_STMT对象
2、将从数据库读取(图片)数据的具体流程如下
1、创建一个MYSQL_STMT *stmt用于表示预处理语句
2、使用mysql_stmt_init()函数初始化MYSQL_STMT结构体对象stmt
3、使用mysql_stmt_prepare()函数将要执行的SQL语句绑定到对象stmt上
4、创建一个结构体MYSQL_BIND *result用于绑定SQL语句中参数的结果集
5、使用mysql_stmt_bind_result(stmt,&result)函数将查询结果绑定到 result 结构体
6、使用mysql_stmt_execute()函数执行SQL语句
7、使用mysql_stmt_store_result()函数将所有结果行保存在本地缓冲区中
8、使用mysql_stmt_fetch()函数获取结果集中行数据并存储到 result.buffer 中
9、使用mysql_stmt_fetch_column() 函数获取结果集列数据,并将其存储到缓冲区的相应位置上
9、在使用完毕后,使用mysql_stmt_close()函数释放MYSQL_STMT对象
#include<stdio.h>
#include<string.h>
#include<mysql.h>#define MING_DB_IP "192.168.42.128"
#define MING_DB_PORT 3306
#define MING_DB_USENAME "admin"
#define MING_DB_PASSWORD "123456"
#define MING_DB_DEFAULTDB "MING_DB"
#define FILE_IMAGE_LENGTH (64*1024)#define SQL_INSERT_IMG_USER "insert TBL_USER(U_NAME,U_GENDER,U_IMG) values('zxm','women',?);"
#define SQL_SELECT_IMG_USER "select U_IMG from TBL_USER where U_NAME='zxm';"//从客户端读取图片存入节点服务器的buffer//filename:图片路径; buffer:存储图片
int read_image(char *filename,char *buffer){if (filename == NULL || buffer ==NULL ) return -1;FILE *fp=fopen(filename,"rb"); //以二进制模式读取该文件if (fp == NULL){printf("fopen failed\n");return -2;}//检测文件大小filefseek(fp,0,SEEK_END);int length=ftell(fp);fseek(fp,0,SEEK_SET);int size=fread(buffer,1,length,fp);if (size != length){printf("fread failed:%d\n",size);return -3;}fclose(fp);return size;
}//从节点服务器的buffer写入图片到客户端
int write_image(char *filename,char *buffer,int length){if (filename == NULL || buffer ==NULL || length <=1) return -1;FILE *fp=fopen(filename,"wb+"); if (fp == NULL){printf("fopen failed\n");return -2;} int size=fwrite(buffer,1,length,fp);if (size != length){printf("fwrite failed:%d\n",size);return -3;}fclose(fp);return size;
}//从节点服务器的buffer读取数据到数据库
int mysql_write(MYSQL *handle,char *buffer,int length){if (handle ==NULL || buffer == NULL || length <=0) return -1;MYSQL_STMT *stmt=mysql_stmt_init(handle);int ret=mysql_stmt_prepare(stmt,SQL_INSERT_IMG_USER,strlen(SQL_INSERT_IMG_USER));if (ret){printf("mysql_stmt_prepare:%s\n",mysql_error(handle));return -2;}MYSQL_BIND param={0};param.buffer_type=MYSQL_TYPE_LONG_BLOB;param.buffer=NULL;param.is_null=0;param.length=NULL;ret=mysql_stmt_bind_param(stmt,¶m);if (ret){printf("mysql_stmt_bind_param:%s\n",mysql_error(handle));return -3; }ret=mysql_stmt_send_long_data(stmt,0,buffer,length);if (ret){printf("mysql_stmt_send_long_data:%s\n",mysql_error(handle));return -4; } ret=mysql_stmt_execute(stmt);if (ret){printf("mysql_stmt_execute:%s\n",mysql_error(handle));return -5; }ret=mysql_stmt_close(stmt) ; if (ret){printf("mysql_stmt_close:%s\n",mysql_error(handle));return -6; } return ret;
}//从数据库写入数据到节点服务器的buffer
int mysql_read(MYSQL *handle,char *buffer,int length){if (handle ==NULL || buffer == NULL || length <=0) return -1;MYSQL_STMT *stmt=mysql_stmt_init(handle);int ret=mysql_stmt_prepare(stmt,SQL_SELECT_IMG_USER,strlen(SQL_SELECT_IMG_USER));if (ret){printf("mysql_stat_prepare:%s\n",mysql_error(handle));return -2;} MYSQL_BIND result={0};result.buffer_type=MYSQL_TYPE_LONG_BLOB;unsigned long total_length=0;result.length=&total_length;ret=mysql_stmt_bind_result(stmt,&result);if (ret){printf("mysql_stmt_bind_result:%s\n",mysql_error(handle));return -3;} ret=mysql_stmt_execute(stmt);if (ret){printf("mysql_stmt_execute:%s\n",mysql_error(handle));return -4;} ret=mysql_stmt_store_result(stmt);if (ret){printf("mysql_stmt_store_result:%s\n",mysql_error(handle));return -5;} while (1){//获取下一行数据并存储到 result.buffer 中ret=mysql_stmt_fetch(stmt);if (ret !=0 && ret != MYSQL_DATA_TRUNCATED ) break;int start=0;while(start <(int)total_length ){/*buffer 是一个指向缓冲区起始地址的指针,而 start 是一个表示偏移量的整数。因此,buffer+start 的结果就是一个指向缓冲区中某个特定位置的指针。result.buffer指向的是buffer+start*/result.buffer=buffer+start; //存储实际读取到的数据长度,单位为字节 result.buffer_length=1;/*在使用 mysql_stmt_fetch_column() 函数从结果集中读取数据时,每次只会读取一部分数据并存储到由result.buffer所指向的缓冲区中。如果要将多个数据段合并成完整的数据块,则需要利用偏移量来调整存储位置。具体地说,可以通过不断修改 start 的值来控制写入数据时所处的位置。*/mysql_stmt_fetch_column(stmt,&result,0,start);start+=result.buffer_length;}}mysql_stmt_close(stmt);return total_length;}int main(){//定义一个MYSQL类型的结构体变量,并通过调用mysql_init()函数对其进行初始化//调用mysql_error()函数来获取有关错误信息MYSQL mysql;if (mysql_init(&mysql)==NULL){printf("mysql_init:%s\n",mysql_error(&mysql));return -1;}//mysql_real_connect()用于连接到MySQL服务器.它需要以下参数:/*1、MYSQL类型的结构体指针,该结构体已由mysql_init()初始化。2、MySQL服务器所在主机名或IP地址。3、登录MySQL服务器时使用的用户名。4、登录MySQL服务器时使用的密码。5、要连接的数据库名称。6、端口号(默认为3306)。7、规定 socket 8、规定不同的连接选项*/if(!mysql_real_connect(&mysql,MING_DB_IP,MING_DB_USENAME,MING_DB_PASSWORD,MING_DB_DEFAULTDB,MING_DB_PORT,NULL,0)){ //等于0,不成功printf("mysql_real_connect:%s\n",mysql_error(&mysql));return -2;}printf("case:mysql read image and write image \n");char buffer[FILE_IMAGE_LENGTH]={0};int length=read_image("/home/zxm/share/06mysqlPicture/0voice.jpg",buffer);if (length < 0) goto Exit;mysql_write(&mysql,buffer,length);printf("case:mysql read mysql and write image \n");memset(buffer,0,FILE_IMAGE_LENGTH);length=mysql_read(&mysql,buffer,FILE_IMAGE_LENGTH);write_image("a.jpg",buffer,length);Exit:mysql_close(&mysql);return 0;}
相关文章:

8、Linux C/C++ 实现MySQL的图片插入以及图片的读取
本文结合了Linux C/C 实现MySQL的图片插入以及图片的读取,特别是数据库读写的具体流程 一、文件读取相关函数 fseek() 可以将文件指针移动到文件中的任意位置。其基本形式如下: int fseek(FILE *stream, long offset, int whence);其中,str…...

【搭建轻量级图床】本地搭建LightPicture开源图床管理系统 - 异地远程访问
文章目录 1.前言2. Lightpicture网站搭建2.1. Lightpicture下载和安装2.2. Lightpicture网页测试2.3.cpolar的安装和注册 3.本地网页发布3.1.Cpolar云端设置3.2.Cpolar本地设置 4.公网访问测试5.结语 1.前言 现在的手机越来越先进,功能也越来越多,而手机…...
微信小程序全局路由拦截
前言 略 微信小程序全局路由拦截方法1 目前微信小程序没有全局路由拦截。要想实现全局路由拦截,需要自己进行扩充。具体参考这里:微信小程序–路由拦截器。 实现思路: 替换Page的参数对象的onShow或onLoad方法。在替换的onShow或onLoad方…...

截图自动添加水印(macOS/windows)
文章目录 1. 截图自动加水印1.1. windows1.2. macOS 2. 对已有图像批量加水印2.1 windows2.2 macOS 1. 截图自动加水印 1.1. windows 直接看这篇文章,一键截图自动生成水印/自动签名主要就是使用一个叫 SPX 的软件 1.2. macOS 其实apple的操作系统,i…...

大学四年,我建议你这么学网络安全
在所有关注我的朋友中,大致分为两类,一类是社会人士,有的是安全老手,有的是其它工作但对安全感兴趣的朋友,另一类应该就是大学生了。 尤其随着国家的号召和知识的普及,越来越多的人开始对网络安全感兴趣&a…...
Spring Boot整合Redis缓存并使用注解
Spring Boot整合Redis缓存并使用注解 在Spring Boot应用程序中,您可以使用Spring Cache库与Redis缓存进行集成,以提高应用程序的性能和响应速度。Spring Cache库提供了一组注解,包括Cacheable、CachePut和CacheEvict,可以方便地将…...
通知可以根据切入点表达式来进行增强,也可以根据自己的注解值来进行增强
通知可以根据切入点表达式来进行增强,也可以根据自己的注解值(例如 Before、After、Around 等)来进行增强。 如果要根据切入点表达式来进行增强,需要在通知注解中使用 Pointcut 注解来引用切入点表达式。例如,在以下示…...

<Python实际应用>做一个简单的签到投屏系统
公司接了个活,承办一个由团委组织的五四青年节徒步活动,其中一个环节是现场报名,来的人把名字填进去后随机分组,并显示在现场的LED大屏幕上,我自告奋勇用Python来开发这个小程序。这里记录一下 【项目需求】 1、报名…...

时序预测 | MATLAB实现BO-CNN-GRU贝叶斯优化卷积门控循环单元时间序列预测
时序预测 | MATLAB实现BO-CNN-GRU贝叶斯优化卷积门控循环单元时间序列预测 目录 时序预测 | MATLAB实现BO-CNN-GRU贝叶斯优化卷积门控循环单元时间序列预测效果一览基本介绍模型描述程序设计参考资料 效果一览 基本介绍 基于贝叶斯(bayes)优化卷积神经网络-门控循环单元(CNN-GR…...
Baumer工业相机堡盟工业相机使用BGAPISDK将工业相机设为Burst模式以及该模式的优势以及行业应用(C++)
Baumer工业相机堡盟工业相机使用BGAPISDK将工业相机设为Burst模式以及该模式的优势以及行业应用(C) Baumer工业相机Baumer工业相机的Burst模式的技术背景Baumer工业相机使用BGAPISDK将设置Burst模式1.引用合适的类文件2.使用BGAPI SDK初始化相机设置Burs…...

BERT输入以及权重矩阵形状解析
以下用形状来描述矩阵。对于向量,为了方便理解,也写成了类似(1,64)这种形状的表示形式,这个你理解为64维的向量即可。下面讲的矩阵相乘都是默认的叉乘。 词嵌入矩阵形状:以BERT_BASE为例,我们知道其有12层Encoder&…...

3 个令人惊艳的 ChatGPT 项目,开源了!
过去一周,AI 界又发生了天翻地覆的变化,其中,最广为人知的,应该是 OpenAI 正式上线 ChatGPT iOS 客户端,让所有人都可以更方便的在手机上与 ChatGPT 聊天。 此外,Stable Diffusion 母公司 Stability AI 也…...
一、12.C++内存管理
C++内存管理 28.C++的内存管理 和C基本一致 代码区(Code Segment):也称为文本区,存放程序的可执行代码。 全局区(Global/Static Segment):存放全局变量、静态变量和常量。程序在编译后,分配这些数据的空间。 栈区(Stack Segment):存放函数调用时的参数、返回地…...

ensp实践dhcp服务
ensp实践dhcp服务 1、dhcp接口分配模式2、dhcp接口地址池分配模式3、dhcp布拉布拉布拉 1、dhcp接口分配模式 1.1、路由器AR1配置dhcp动态获取 <Huawei>system-view [Huawei]interface g0/0/0 [Huawei-GigabitEthernet0/0/0]ip address 10.1.1.1 24 [Huawei-GigabitEthe…...

【王道·计算机网络】第六章 应用层
一、基本概念 1.1 应用层概述 应用层对应用程序的通信提供服务应用层协议定义: 应用进程交换的报文类型,请求还是响应?各种报文类型的语法,如报文中的各个字段及其详细描述字段的语义,即包含在字段中的信息的含义进程何时、如何…...

【论文解读】(如何微调BERT?) How to Fine-Tune BERT for Text Classification?
文章目录 论文信息1. 论文内容2. 论文结论2.1 微调流程2.2 微调策略(Fine-Tuning Strategies)2.3 Further Pretrain 3. 论文实验介绍3.1 实验数据集介绍3.2 实验超参数3.3 Fine-Tuning策略探索3.3.1 处理长文本3.3.2 不同层的特征探索3.3.3 学习率探索(灾难性遗忘探…...

工程师是怎样对待开源
工程师如何对待开源 本文是笔者作为一个在知名科技企业内从事开源相关工作超过 20 年的工程师,亲身经历或者亲眼目睹很多工程师对待开源软件的优秀实践,也看到了很多 Bad Cases,所以想把自己的一些心得体会写在这里,供工程师进行…...

Spring Boot日志系统大揭秘:从零开始学习Spring Boot日志:常见问题解答和最佳实践
一. 关于 Spring Boot 日志的使用 Spring Boot 日志机制和工具用于记录应用程序的日志信息和追踪应用程序的执行过程。它集成了常用的日志框架,如 Log4j、logback、Java Util Logging等,并提供简单易用的配置方式,让开发人员可以方便地监控应…...
【06】Nginx之反向代理
Nginx反向代理的配置语法 Nginx反向代理模块的指令是由ngx_http_proxy_module模块进行解析 proxy_pass 该指令用来设置被代理服务器地址,可以是主机名称、IP地址加端口号形式。 语法proxy_pass URL;默认值—位置location URL:为要设置的被代理服务器地址…...

TCP是面向字节流的协议
TCP字节流 之所以会说 TCP 是面向字节流的协议,UDP 是面向报文的协议,是因为操作系统对 TCP 和 UDP 协议的发送方的机制不同,也就是问题原因在发送方。 为什么 UDP 是面向报文的协议? 当用户消息通过 UDP 协议传输时,…...
Objective-C常用命名规范总结
【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名(Class Name)2.协议名(Protocol Name)3.方法名(Method Name)4.属性名(Property Name)5.局部变量/实例变量(Local / Instance Variables&…...
生成 Git SSH 证书
🔑 1. 生成 SSH 密钥对 在终端(Windows 使用 Git Bash,Mac/Linux 使用 Terminal)执行命令: ssh-keygen -t rsa -b 4096 -C "your_emailexample.com" 参数说明: -t rsa&#x…...
【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】
1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...
数据库分批入库
今天在工作中,遇到一个问题,就是分批查询的时候,由于批次过大导致出现了一些问题,一下是问题描述和解决方案: 示例: // 假设已有数据列表 dataList 和 PreparedStatement pstmt int batchSize 1000; // …...

dify打造数据可视化图表
一、概述 在日常工作和学习中,我们经常需要和数据打交道。无论是分析报告、项目展示,还是简单的数据洞察,一个清晰直观的图表,往往能胜过千言万语。 一款能让数据可视化变得超级简单的 MCP Server,由蚂蚁集团 AntV 团队…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”
2025年#高考 将在近日拉开帷幕,#AI 监考一度冲上热搜。当AI深度融入高考,#时间同步 不再是辅助功能,而是决定AI监考系统成败的“生命线”。 AI亮相2025高考,40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕,江西、…...

RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)
RabbitMQ 一、RabbitMQ概述 RabbitMQ RabbitMQ最初由LShift和CohesiveFT于2007年开发,后来由Pivotal Software Inc.(现为VMware子公司)接管。RabbitMQ 是一个开源的消息代理和队列服务器,用 Erlang 语言编写。广泛应用于各种分布…...

宇树科技,改名了!
提到国内具身智能和机器人领域的代表企业,那宇树科技(Unitree)必须名列其榜。 最近,宇树科技的一项新变动消息在业界引发了不少关注和讨论,即: 宇树向其合作伙伴发布了一封公司名称变更函称,因…...
前端中slice和splic的区别
1. slice slice 用于从数组中提取一部分元素,返回一个新的数组。 特点: 不修改原数组:slice 不会改变原数组,而是返回一个新的数组。提取数组的部分:slice 会根据指定的开始索引和结束索引提取数组的一部分。不包含…...

AxureRP-Pro-Beta-Setup_114413.exe (6.0.0.2887)
Name:3ddown Serial:FiCGEezgdGoYILo8U/2MFyCWj0jZoJc/sziRRj2/ENvtEq7w1RH97k5MWctqVHA 注册用户名:Axure 序列号:8t3Yk/zu4cX601/seX6wBZgYRVj/lkC2PICCdO4sFKCCLx8mcCnccoylVb40lP...