C语言连接【MySQL】

稍等更新图片。。。。

文章目录

  • 安装 MySQL 库
  • 连接 MySQL
    • MYSQL 类
    • 创建 MySQL 对象
    • 连接数据库
    • 关闭数据库连接
    • 示例
  • 发送命令
  • 设置编码格式
  • 插入、删除或修改记录
  • 查询记录
    • 示例
  • 参考资料

安装 MySQL 库

在 CentOS7 下,使用命令安装 MySQL:

yum install mysql-devel

/usr/include可以看到一个mysql新目录,里面存放的是 mysql 的头文件。另外在 /lib64/mysql/ 以及 /usr/lib64/mysql 目录下存放了 mysql 的动态和静态库。

用一个 MySQL 库提供的接口验证 MySQL 库是否安装成功:

#include <iostream>
#include <mysql/mysql.h>

using namespace std;

int main()
{
	cout << "mysql version: " << mysql_get_client_info() << endl;
	return 0;
}

编译:

g++ sql.cc -o sql -I/usr/include/mysql -L/usr/lib64/mysql -lmysqlclient
image-20240227182356233

编译选项中关于库的使用:

  • -I:用于指明头文件的搜索路径。
  • -L:用于指明库文件的搜索路径。
  • -l:用于指明需要连接库文件路径下的哪一个库。
image-20240227182921177

因为这个库没有在链接的默认目录/usr/lib64下,所以作为第三方导入的库,在编译时需要显式地指定-L/usr/lib64/mysql;同理,头文件不在默认目录/usr/include下,所以要显式地指定-I/usr/include/mysql。在这个目录下,存在名为mysqlclient的第三方库,同样需要用-l显式地声明。

只要正常运行上面的程序,那就表明库的链接没有问题,剩下的就是简单的 API 使用。

连接 MySQL

MYSQL 类

在使用 MySQL 提供的接口之前,需要了解一下这个重要的类。

MYSQL类是一个非常核心的结构体,它用于表示与 MySQL 服务器的一个连接实例。在客户端程序中,这个结构体用来保存客户端与数据库服务器之间连接的所有必要信息,包括但不限于:

  • 服务器的地址
  • 用户名和密码
  • 正在使用的数据库
  • 网络连接的状态和配置
  • 错误信息和错误码
  • 查询结果
  • 选项设置

mysql.h中可以查看 MYSQL 结构体的定义(了解即可):

typedef struct st_mysql
{
  NET		net;			/* Communication parameters */
  unsigned char	*connector_fd;		/* ConnectorFd for SSL */
  char		*host,*user,*passwd,*unix_socket,*server_version,*host_info;
  char          *info, *db;
  struct charset_info_st *charset;
  MYSQL_FIELD	*fields;
  MEM_ROOT	field_alloc;
  my_ulonglong affected_rows;
  my_ulonglong insert_id;		/* id if insert on table with NEXTNR */
  my_ulonglong extra_info;		/* Not used */
  unsigned long thread_id;		/* Id for connection in server */
  unsigned long packet_length;
  unsigned int	port;
  unsigned long client_flag,server_capabilities;
  unsigned int	protocol_version;
  unsigned int	field_count;
  unsigned int 	server_status;
  unsigned int  server_language;
  unsigned int	warning_count;
  struct st_mysql_options options;
  enum mysql_status status;
  my_bool	free_me;		/* If free in mysql_close */
  my_bool	reconnect;		/* set to 1 if automatic reconnect */

  /* session-wide random string */
  char	        scramble[SCRAMBLE_LENGTH+1];
  my_bool unused1;
  void *unused2, *unused3, *unused4, *unused5;

  LIST  *stmts;                     /* list of all statements */
  const struct st_mysql_methods *methods;
  void *thd;
  /*
    Points to boolean flag in MYSQL_RES  or MYSQL_STMT. We set this flag 
    from mysql_stmt_close if close had to cancel result set of this object.
  */
  my_bool *unbuffered_fetch_owner;
  /* needed for embedded server - no net buffer to store the 'info' */
  char *info_buffer;
  void *extension;
} MYSQL;
  • MYSQL 对象中的 methods 成员是一个结构体变量,该变量里面保存着很多函数指针,这些函数指针将会在数据库连接成功以后的各种数据操作中被调用。

创建 MySQL 对象

MYSQL* mysql_init(MYSQL *mysql);
  • 该函数用来分配或者初始化一个 MySQL 对象,用于连接 MySQL 服务器。
  • 如果传入的参数是 NULL,那么 mysql_init 将自动为你分配一个 MySQL 对象并返回。
  • 如果传入的参数是一个地址,那么 mysql_init 将在该地址处帮你完成初始化。

连接数据库

MYSQL* mysql_real_connect(MYSQL *mysql, const char *host,
					const char *user,
					const char *passwd,
					const char *db,
					unsigned int port,
					const char *unix_socket,
					unsigned long clientflag);

其中:

  • mysql: 表示在连接数据库前,调用 mysql_init 函数创建的 MySQL 对象。
  • host: 表示需要连接的 MySQL 服务器的 IP 地址,"127.0.0.1"表示连接本地 MySQL 服务器。
  • user: 表示连接 MySQL 服务器时,所使用用户的用户名。
  • passwd: 表示连接 MySQL 服务器时,所使用用户的密码
  • db: 表示连接 MySQL 服务器后,需要使用的数据库。
  • port: 表示连接的 MySQL 服务器,所对应的端口号。
  • unix_socket: 表示连接时应该使用的套接字或命名管道,通常设置为 NULL。
  • clientflag: 可以设置为多个标志位的组合,表示允许特定的功能,通常设置为 0。

返回值说明:

  • 如果连接数据库成功,则返回一个 MySQL 对象,该对象与第一个参数的值相同。
  • 如果连接数据库失败,则返回 NULL。

关闭数据库连接

void mysql_close(MYSQL *sock);

其中:

  • 该函数的参数,就是连接数据库前调用 mysql_init 创建的 MySQL 对象。
  • 如果传入的 MySQL 对象是 mysql_init 自动创建的,那么调用 mysql_close 时就会释放这个对象。

示例

在 MySQL 中首先有一个新用户:

grant all on curd_db.* to 'new_user'@'%' identified by '12345';

用户名是new_user%表示任意主机的用户,grant all表示它被授予所有权限在curd.db数据库下,密码是12345

在本地测试:

#include <iostream>
#include <string>
#include <mysql/mysql.h>
using namespace std;

const string host = "localhost";
const string user = "new_user";
const string passwd = "12345";
const string db = "curd_db";
const int port = 3306;

int main()
{
	// 1、创建 MySQL 对象
	MYSQL *mySQL = mysql_init(nullptr);
	// 2、连接数据库
	if (mysql_real_connect(mySQL, host.c_str(), user.c_str(), passwd.c_str(), db.c_str(), port, nullptr, 0) == nullptr)
	{
		cerr << "数据库连接失败!" << endl;
		exit(1);
	}
	cout << "数据库连接成功!" << endl;

	// 3、关闭数据库
	mysql_close(mySQL);
	cout << "数据库关闭成功!" << endl;

	return 0;
}

编译并运行:

image-20240227190559910

发送命令

int mysql_query(MYSQL *mysql, const char *stmt_str);

数用于向 MySQL 服务器发送一个查询或命令,执行指定的 SQL 语句。

参数说明:

  • MYSQL *mysql:指向 MYSQL 结构体的指针,这个结构体代表了与 MySQL 服务器的一个连接。
  • const char *stmt_str:要执行的 SQL 语句的字符串。

返回值:

  • 成功执行时,返回 0
  • 出现错误时,返回非 0 值。

设置编码格式

在连接数据库之后,需要统一客户端和服务器的编码格式,避免在数据交互过程中出现乱码,设置编码格式的函数如下:

int mysql_set_character_set(MYSQL *mysql, const char *csname);

参数说明:

  • mysql: 表示在连接数据库前,调用 mysql_init 函数创建的 MySQL 对象。
  • csname: 表示要设置的编码格式,如"utf8"

返回值说明:

  • 返回值为 0 表示设置成功,否则表示设置失败。

插入、删除或修改记录

在 mysql_query 函数中向 MySQL 发送 INSERT SQL:

int main()
{
    // ...
	cout << "数据库连接成功!" << endl;
	// 设置编码
	mysql_set_character_set(mySQL, "utf8");
	// 插入记录
	string sql = "insert into account values(4,'小李',30,400)";
	if (mysql_query(mySQL, sql.c_str()) != 0)
	{
		cout << "插入数据失败!" << endl;
		exit(2);
	}
	// ...
	return 0;
}

数据被成功插入到表中。

image-20240227193308014

类似地,可以删除和修改记录:

	string sql = "update account set balance=222 where id=2";
image-20240227193749866
	string sql = "delete from account where id=4";
image-20240227193844242

查询记录

对于 mysql_query 函数而言,插入、删除和修改操作都很简单,只要将 SQL 字符串作为参数传入即可,不需要返回值。但是 SELECT 查询需要返回结果,这需要使用到 MYSQL_RES 对象。

MYSQL_RES* mysql_store_result(MYSQL *mysql);
  • 该函数会调用指定 MySQL 对象中对应的函数指针来获取查询结果,并将获取到的查询结果保存到 MYSQL_RES 变量中进行返回。
  • 需要注意的是,MYSQL_RES 变量的内存空间是 malloc 出来的,因此在使用完后需要调用 free 函数进行释放,否则会造成内存泄露。

MYSQL_RES 变量中保存了查询得到的各种信息,其类型定义如下:

typedef struct st_mysql_res {
	my_ulonglong  row_count;
	MYSQL_FIELD	*fields;
	MYSQL_DATA	*data;
	MYSQL_ROWS	*data_cursor;
	unsigned long *lengths;		/* column lengths of current row */
	MYSQL		*handle;		/* for unbuffered reads */
    const struct st_mysql_methods *methods;
    MYSQL_ROW	row;			/* If unbuffered read */
    MYSQL_ROW	current_row;		/* buffer to current row */
    MEM_ROOT	field_alloc;
    unsigned int	field_count, current_field;
    my_bool	eof;			/* Used by mysql_fetch_row */
    /* mysql_stmt_close() had to cancel this result */
    my_bool       unbuffered_fetch_cancelled;
    void *extension;
} MYSQL_RES;

获取查询结果的行数:

my_ulonglong mysql_num_rows(MYSQL_RES *res);

获取查询结果的列数:

unsigned int mysql_num_fields(MYSQL_RES *res);

获取查询结果的列属性:

MYSQL_FIELD* mysql_fetch_fields(MYSQL_RES *res);

mysql_fetch_fields 函数将会返回多个 MYSQL_FIELD 对象,每个 MYSQL_FIELD 对象中保存着对应列的各种列属性,其类型定义如下:

typedef struct st_mysql_field {
	char *name;                 /* Name of column */
	char *org_name;             /* Original column name, if an alias */
	char *table;                /* Table of column if column was a field */
	char *org_table;            /* Org table name, if table was an alias */
	char *db;                   /* Database for table */
	char *catalog;	      /* Catalog for table */
	char *def;                  /* Default value (set by mysql_list_fields) */
	unsigned long length;       /* Width of column (create length) */
	unsigned long max_length;   /* Max width for selected set */
	unsigned int name_length;
	unsigned int org_name_length;
	unsigned int table_length;
	unsigned int org_table_length;
	unsigned int db_length;
	unsigned int catalog_length;
	unsigned int def_length;
	unsigned int flags;         /* Div flags */
	unsigned int decimals;      /* Number of decimals in field */
	unsigned int charsetnr;     /* Character set */
	enum enum_field_types type; /* Type of field. See mysql_com.h for types */
	void *extension;
} MYSQL_FIELD;

获取查询结果中的一行数据:

MYSQL_ROW mysql_fetch_row(MYSQL_RES *result);

MYSQL_ROW 对象中保存着一行数据,这一行数据中可能包含多个字符串,对应就是这行数据中的多个列信息,因此 MYSQL_ROW 本质就是 char** 类型,其类型定义如下:

typedef char **MYSQL_ROW;		/* return data as array of strings */

示例

int main()
{
    // ...
	// 3、查询数据库表中的记录
	// a、执行查询语句
	string sql = "select * from account";
	if (mysql_query(mySQL, sql.c_str()) != 0)
	{
		cout << "查询数据失败!" << endl;
		exit(2);
	}
	cout << "查询数据成功!" << endl;
	// b、获取查询结果
	MYSQL_RES *res = mysql_store_result(mySQL);
	int rows = mysql_num_rows(res);	  // 数据的行数
	int cols = mysql_num_fields(res); // 数据的列数
	// 获取每列的属性并打印列名
	MYSQL_FIELD *fields = mysql_fetch_fields(res);
	for (int i = 0; i < cols; i++)
	{
		cout << fields[i].name << "\t";
	}
	cout << endl;
	for (int i = 0; i < rows; i++)
	{
		// 获取一行数据并进行打印
		MYSQL_ROW row = mysql_fetch_row(res);
		for (int j = 0; j < cols; j++)
		{
			cout << row[j] << "\t";
		}
		cout << endl;
	}
	// 释放内存空间
	free(res); 
    // ...
	return 0;
}

参考资料

  • Linux centos 7/ubantu 下: 用 C 语言连接 MySQL 数据库
  • MySQL 使用 C 语言连接

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/446989.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

arcgis栅格数据处理3——定义投影(同样适用于其他类型文件)

进行数据连接时可能出现未设置投影无法链接的情况&#xff0c;需要先定义投影 点击最右侧“目录”&#xff0c;弹出带有系统工具的面板&#xff0c;点击“data management tools”点击“投影”&#xff0c;“定义投影”

【轮式平衡机器人】——TMS320F28069片内外设之eCAP

引入 TMS320F28069的eCAP&#xff08;增强型捕获模块&#xff09;是一个强大的外设&#xff0c;用于精确测量和捕获输入信号的事件和时间戳。 在电机控制、传感器数据采集和信号处理等应用中&#xff0c;eCAP模块可以用于测量霍尔传感器、编码器或其他数字输入信号的周期、频…

计算表达式x*(2^i)的值math.ldexp(x, i)

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 计算表达式x*(2^i)的值 math.ldexp(x, i) [太阳]选择题 关于以下代码输出的结果说法正确的是&#xff1f; import math print("【执行】math.ldexp(3,2)") print(math.ldexp(3,2)) …

2024/3/10总结:数据结构教程:顺序表的创建以及基本的12个操作

首先&#xff0c;按照惯例&#xff0c;欢迎大家边听歌边看本博客&#xff01;&#xff01;&#xff01; 这里是神奇的赛尔号_张杰 (kugou.com) 一.背景&#xff1a;由于是上机实验&#xff0c;直接引用数据结构教程第6版73页的实验题1 修改第6&#xff0c;7&#xff0c;8&am…

CI/CD笔记.Gitlab系列:控制台强制修改root用户密码

CI/CD笔记.Gitlab系列 控制台强制修改root用户密码 - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article:https://blog.cs…

SpringBoot中的上传文件接口

SpringBoot中的上传文件 上传文件的操作在有些功能中属于比较常用的环节&#xff0c;这里整理下SpringBoot环境中上传文件的实现方式。 这里实现的是上传文件的后台接口&#xff0c;前端部分可以用测试工具模拟实现&#xff0c;就先不在这里表述了。 Dto层 使用MultipartFile…

【C++】类和对象(六个默认成员函数)

文章目录 类的六个默认成员函数**构造函数****构造函数的目的****构造函数的特性** 析构函数析构函数概念析构函数处理的顺序析构函数清理细节 拷贝构造函数拷贝构造函数典型调用场景 赋值运算符重载运算符重载赋值运算重载前置和后置 重载 const成员函数再提权限的问题: 取地址…

Guiding Large Language Models viaDirectional Stimulus Prompting

1. 通过定向刺激提示指导大语言模型 论文地址&#xff1a;[2302.11520] Guiding Large Language Models via Directional Stimulus Prompting (arxiv.org) 源码地址&#xff1a;GitHub - Leezekun/Directional-Stimulus-Prompting: [NeurIPS 2023] Codebase for the paper: &qu…

目标检测论文模型笔记——RCNN系列

RCNN系列模型&#xff08;two-stages、基于区域的)主要包括以下几种&#xff0c;按发布时间排序&#xff1a; RCNN&#xff08;2014年&#xff09;&#xff1a;首次将深度学习应用于目标检测&#xff0c;通过选择性搜索Selective Search提出候选区域&#xff0c;然后使用CNN&am…

章六、集合(1)—— 概念、API、List 接口及实现类、集合迭代

零、 关闭IDEA调试时自动隐藏空元素 一、 集合的概念 存储一个班学员信息&#xff0c;假定一个班容纳20名学员 当我们需要保存一组一样&#xff08;类型相同&#xff09;的元素的时候&#xff0c;我们应该使用一个容器来存储&#xff0c;数组就是这样一个容器。 数组有什么缺…

9. 内核、文件系统加载工具

内核、文件系统加载工具 内核、文件系统加载工具是嵌入式开发必备的工具 1. 烧写BootLoader 1.1 通过超级终端方式 烧写 Bootloader 可以使用超级终端的“传送” |“发送文件”命令进入发送文件对话框&#xff0c;使用 Xmodem 协议和 Kermit 协议发送 Bootloader 的各个文件…

《计算机网络》考研:2024/3/9 2.1.7-数据交换方式;2.2-物理层传输介质;2.3-物理层设备

2024/3/9 2.1.7、2.2、2.3 2.1.7 数据交换方式 电路交换存储转发方式 报文交换分组交换&#xff1a; 数据报方式虚电路方式 电路交换 报文交换 分组交换 2.2 物理层传输介质 物理层的主要任务 物理层设备 中继器&#xff1a; 集线器&#xff08;多口中继器&#xff09;…

如何获取用户请求的真实ip,并返回访问者的ip地理位置?node,vue

一、获取真实IP 方式1、前端调用免费公共接口获取 前端获取访问者的真实的外网ip,可以通过调用接口https://api.ipify.org/来获取。你也可以直接在网页上访问它来看自己的外网ip。 ipify介绍&#xff1a; ipify是一个免费的公共 API&#xff0c;用于获取设备的公共 IP 地址。…

Claude3横空出世:颠覆GPT-4,Anthropic与亚马逊云科技共启AI新时代

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

exceljs解析和生成excel文件

安装 npm install exceljs解析excel 通过 Workbook 的 readFile 方法可以拿到workbook对象, workbook对象包含的概念有 worksheet(工作表) --> row(行) --> cell(单元格).于是可以通过依次遍历 worksheet, row, cell来拿到单元格的数据直接通过 worksheet.getSheetValue…

从零学习Linux操作系统 第三十五部分 Ansible中的角色

一、理解roles在企业中的定位及写法 #ansible 角色简介# Ansible roles 是为了层次化&#xff0c;结构化的组织Playbookroles就是通过分别将变量、文件、任务、模块及处理器放置于单独的目录中&#xff0c;并可以便捷地include它们roles一般用于基于主机构建服务的场景中&…

Springboot 集成kafka 消费者实现ssl方式连接监听消息实现消费

证书准备&#xff1a;springboot集成kafka 消费者实现 如何配置是ssl方式连接的时候需要进行证书的转换。原始的证书是pem, 或者csr方式 和key方式的时候需要转换&#xff0c;因为kafka里面是jks 需要通过openssl进行转换。 证书处理&#xff1a; KeyStore 用于存储客户端的证…

Java多线程实战-实现多线程文件下载,支持断点续传、日志记录等功能

&#x1f3f7;️个人主页&#xff1a;牵着猫散步的鼠鼠 &#x1f3f7;️系列专栏&#xff1a;Java全栈-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 目录 前言 1 基础知识回顾 1.1 线程的创建和启动 1.2 线程池的使用 2.运行环境说…

k8s架构浅析

Node 节点&#xff08;物理主机或虚拟机&#xff09;&#xff0c;它们共同组成一个分布式集群&#xff0c;并且这些节点中会有一个 Master 节点&#xff0c;由它来统一管理 Node 节点。 Pod &#xff0c;在 K8S 中&#xff0c;Pod 是最基本的操作单元&#xff0c;它与 docker …

Linux之selinux详解

华子目录 概念作用selinux与传统的权限区别selinux工作原理名词解释主体&#xff08;subject&#xff09;目标&#xff08;object&#xff09;策略&#xff08;policy&#xff09;&#xff08;多个规则的集合&#xff09;安全上下文&#xff08;security context&#xff09; 文…