当前位置: 首页 > news >正文

【MySQL】Linux下用C/C++链接MySQL数据库

文章目录

  • 一、准备工作
  • 二、验证库和接口的使用
  • 三、链接数据库
  • 四、对数据库进行增删查改
  • 五、结尾


一、准备工作

要使用C++链接数据库, 首先要去MySQL官网下载官网提供的库, MySQL 社区下载.
如图所示:
在这里插入图片描述
接着选择:
在这里插入图片描述
按需选择版本:
在这里插入图片描述
如果用的是云服务器, 那么在安装mysql时默认就会把这些库都给装上了.

成功安装后会获得这些库和接口:
查看头文件指令:

ls /usr/include/mysql/

在这里插入图片描述
查看库的指令:

ls /lib64/mysql/

在这里插入图片描述
如果查看对应目录存在这些库和接口就ok了.

二、验证库和接口的使用

需要包含的头文件: #include <mysql/mysql.h>
测试代码:

#include <mysql/mysql.h>
#include <iostream>
using namespace std;int main()
{cout << "mysql client version: " << mysql_get_client_info() << endl;return 0;
}

函数: mysql_get_client_info()
这个函数的功能就是查看当前mysql客户端的版本, 如果能成功执行并输出版本号, 就表示当前库和接口都是可以使用的, 可以开始链接数据库了.
一般来说直接编译会失败, 会出现类似提示:
在这里插入图片描述
要解决该问题, 首先在编译时指明头文件所在目录, 库所在目录, 再指定需要链接的库的名字, 如下:

g++ -o test test.cpp -std=c++11 -I /usr/include/mysql -L /lib64/mysql -lmysqlclient

再把库所在的路径添加到系统的动态库查找路径中:

export LD_LIBRARY_PATH=./lib64

此时直接执行就好了, 结果如下:
在这里插入图片描述
成功输出版本信息就对了.

三、链接数据库

  1. 首先需要定义一个类型为MYSQL*的变量, 并使用MYSQL *mysql_init(MYSQL *mysql)函数对该变量进行初始化
MYSQL* my = mysql_init(nullptr); //定义并初始化MySQL变量
  1. 初始化完毕后, 进行链接数据库操作, 通过函数:
MYSQL* mysql_real_connect(MYSQL *mysql, //调用mysql_init()以初始化MYSQL结构变量const char *host, //主机名或 IP 地址const char *user, //用户的 MySQL 登录 IDconst char *passwd, //密码, 如果passwd为NULL,则仅检查表中user具有空白密码字段的用户条目是否匹配。const char *db, //数据库名称unsigned int port, //端口号const char *unix_socket, //如果unix_socket不是 NULL,则字符串指定要使用的套接字或命名管道unsigned long client_flag //client_flag通常为0,但可以设置为其他标志的组合以启用某些功能)

示例代码:

mysql_real_connect(my, "127.0.0.1", "root", nullptr, "conn", 3306, nullptr, 0)

关于mysql_real_connect()的返回值:
在这里插入图片描述
可以根据返回结果判断是否真的链接上数据库, 具体代码:

#include <mysql/mysql.h>
#include <iostream>
using namespace std;int main()
{MYSQL* my = mysql_init(nullptr); //定义并初始化MySQL变量//链接数据库if(mysql_real_connect(my, "127.0.0.1", "root", nullptr, "conn", 3306, nullptr, 0)){cout << "connect success." << endl;}return 0;
}

结果:
在这里插入图片描述

四、对数据库进行增删查改

向链接的数据库下发sql语句的函数int mysql_query(MYSQL *mysql, const char *stmt_str).
第一个参数是之前初始化的MYSQL*变量, 第二个参数是字符串形式的sql语句.
返回值:
在这里插入图片描述

增删查改操作的表结构:
在这里插入图片描述

首先看到表里是没有数据的:
在这里插入图片描述
通过mysql_query()插入数据:

mysql_query(my, "insert info(name,age,gender) values('张三',18,1)");

执行后, 数据成功插入对应的数据库中:
在这里插入图片描述
其他的数据都和预期插入的数据一致, 唯独name不一致, 这是因为我的数据库的字符集是utf8而通过语言设置链接的默认字符集为latin1, 由于字符集不匹配而导致了乱码, 所以在建立链接后可以修改默认字符集:

mysql_set_character_set(my, "utf8"); //设置字符集

再编译执行:
在这里插入图片描述
此次插入的数据就没问题了.

通过

mysql_query(my, "delete from info;");

直接执行, 结果:
在这里插入图片描述
成功删除.

现在表中没有数据, 先插入一条数据:

mysql_query(my, "insert info values(1,'张三',18,1)");

查看表中的数据:
在这里插入图片描述
接着输入更新的sql语句就完事了:

mysql_query(my, "update info set age=28 where id=1");

将年龄更改为28, 编译执行后:
在这里插入图片描述

增删改都很简单, 就是在mysql_query()中写sql语句, 然后查看操作是否成功就好了, 但是查的话不太一样, 虽然查询语句也是直接在mysql_query()中写, 但是查的关键是要拿到查询结果, 需要获得查询结果可以通过函数:

MYSQL_RES* mysql_store_result(MYSQL *mysql)

参数就是之前定义的那个MYSQL类型的变量, 而返回值是重点:
在这里插入图片描述
也就是要定义一个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_query(my, "select * from info"); //查询MYSQL_RES* res = mysql_store_result(my); //获取结果结构体
int row = mysql_num_rows(res); //获取行数
int col = mysql_num_fields(res); //获取列数
MYSQL_FIELD* fields = mysql_fetch_fields(res); //获得属性名称for(int i = 0; i < col; ++i)
{cout << fields[i].name << " ";
}
cout << endl;

输出结果:
在这里插入图片描述
没问题, 成功拿到属性名称, 接着来获取数据内容, 通过函数MYSQL_ROW mysql_fetch_row(MYSQL_RES *result)可以获取到, 其返回值为MYSQL_ROW类型的一个变量, 可以把它当成一个char**的二维数组来使用, 类似:
在这里插入图片描述
例子:

MYSQL_ROW line;
for(int i = 0; i < row; ++i)
{line = mysql_fetch_row(res);for(int j = 0; j < col; ++j){cout << line[j] << '\t';}cout << endl;
}

输出结果:
在这里插入图片描述
没问题, 成功获取到了表中的内容数据, 需要注意想要获取下一行数据内容并不需要我们手动操作, 每调用一次mysql_fetch_row(res)函数都会自动获取下一行的内容数据, 然后再遍历该行的内容数据即可, 完整获取查询结果代码:

mysql_query(my, "select * from info");MYSQL_RES* res = mysql_store_result(my); //获取结果结构体
int row = mysql_num_rows(res); //获取行数
int col = mysql_num_fields(res); //获取列数
MYSQL_FIELD* fields = mysql_fetch_fields(res); //获得属性名称//获取属性名称
for(int i = 0; i < col; ++i)
{cout << fields[i].name << '\t';
}
cout << endl;MYSQL_ROW line;
//获取内容数据
for(int i = 0; i < row; ++i)
{line = mysql_fetch_row(res);for(int j = 0; j < col; ++j){cout << line[j] << '\t';}cout << endl;
}

结果:
在这里插入图片描述

五、结尾

在最后不需要访问数据库了, 记得把链接关了, 通过函数void mysql_close(MYSQL *mysql)来关闭链接, 参数即为之前创建的MYSQL*类型的变量.
如需了解其他接口, 可直接访问MySQL的官方文档: C API 函数参考

相关文章:

【MySQL】Linux下用C/C++链接MySQL数据库

文章目录 一、准备工作二、验证库和接口的使用三、链接数据库四、对数据库进行增删查改增删改查 五、结尾 一、准备工作 要使用C链接数据库, 首先要去MySQL官网下载官网提供的库, MySQL 社区下载. 如图所示: 接着选择: 按需选择版本: 如果用的是云服务器, 那么在安装mysql时…...

Python金融量化专栏简介

量化分析实战 - 专栏大纲 👉👉👉 《玩转Python金融量化专栏》👈👈👈 订阅本专栏的可以下载对应的代码和数据集 专栏目标 本专栏旨在帮助读者全面掌握使用Python进行金融技术指标的计算与应用,从基础到高级,涵盖各种技术指标的实现、策略开发与回测等内容。通过…...

出行365:依托分布式数据库,让出行无忧 | OceanBase案例

*本文首发自“新华社环球”杂志&#xff0c;作者张海鑫 每年的暑期旅游旺季&#xff0c;都会触发一轮轮的文旅消费的热潮&#xff0c;对于互联网出行服务行业而言&#xff0c;这既是一场盛大的狂欢&#xff0c;也是对其综合实力的严峻考验。 然而&#xff0c;自去年暑假起&…...

【C语言】位段详解

&#x1f984;个人主页:小米里的大麦-CSDN博客 &#x1f38f;所属专栏:https://blog.csdn.net/huangcancan666/category_12718530.html &#x1f381;代码托管:黄灿灿 (huang-cancan-xbc) - Gitee.com ⚙️操作环境:Visual Studio 2022 目录 一、什么是位段&#xff1f; 二、…...

LVS集群实验

NAT模式 本质是多目标IP的DNAT&#xff0c;通过将请求报文中的目标地址和目标端口修改为某挑出的RS的RIP和PORT实现转发RIP和DIP应在同一个IP网络&#xff0c;且应使用私网地址:RS的网关要指向DIP请求报文和响应报文都必须经由Direclor转发&#xff0c;Direclor易于成为系统瓶…...

在 Spring Boot 中使用适配器模式实现支付网关的统一接口

引言 在许多电子商务系统中&#xff0c;集成多个支付网关是常见的需求。不同的支付网关有着不同的接口和实现细节。适配器模式可以帮助我们以一种灵活的方式实现这些不同的支付网关接口。 适配器模式简介 适配器模式将一个类的接口转换为客户期望的另一个接口。适配器模式使…...

【书生·浦语大模型实战营】第三期 入门岛作业

入门岛作业 Linux闯关任务&#xff1a;完成 SSH 连接与端口映射并运行 hello_world.py。配置vscode作业内容 可选任务1&#xff1a;将Linux基础命令在开发机上完成一遍作业内容 可选任务 2&#xff1a;使用 VSCODE 远程连接开发机并创建一个conda环境作业内容 可选任务 3&#…...

Redis的String类型常用命令总结

1. set 设置一个键的值。 set key value示例&#xff1a; set username "alice"2. get 获取一个键的值。 get key示例&#xff1a; get username3. getset 设置键的值&#xff0c;并返回键的旧值。 getset key value示例&#xff1a; getset username "…...

河南萌新联赛2024第(四)场:河南理工大学

A 思路&#xff1a; B 思路&#xff1a;有一种贪心的写法&#xff0c;将整个数组排序以后比较两个相邻数的同或值&#xff0c;取 m a x max max&#xff0c;不会证明 int th(int x, int y, int z) {int res 0;for (int i z - 1; i > 0; i --) {int dx (x >> i &…...

Linux中临时使用账号提权进行业务操作

普通账号提权 su&#xff1a;永久提权 Switching users with su sudo&#xff1a;临时提权 Running commands as root with sudo 1&#xff09;su切换账号 需要对方的密码 示例&#xff1a;切换到超级管理员 su - root 建议用-这个 su root 普通用户需要执行特殊指令&…...

lwip 3. 网线拔掉后 lwip_recvfrom不能返回

当网线被拔掉后&#xff0c;‌LWIP的lwip_recvfrom函数无法返回&#xff0c;‌这通常意味着网络连接已经断开&#xff0c;‌而LWIP没有自动检测到这种断开并进行相应的处理。具体卡在这个地方&#xff1a; ret xQueueReceive(mbox->mbx, &(*msg), portMAX_DELAY); //具…...

Linux环境安装Docker Engine

Docker是一个开源的应用容器引擎&#xff0c;由Go语言开发&#xff0c;基于Linux内核技术。Docker通过将应用及其依赖打包到可移植的容器中&#xff0c;实现了应用的快速部署和高效管理。Docker容器具有轻量级、快速启动、可移植性强等特点&#xff0c;能够显著提升资源利用率和…...

大厂面试题分享

大厂面试题分享 Redis持久化方式AOF优缺点RDB优缺点 如何保证Redis和Myql的一致性索引下推输入url到浏览器发生了什么ReentranLock底层原理SpringBoot 的启动流程 Redis持久化方式 Redis提供了两种主要的持久化机制&#xff0c;分别是AOF&#xff08;Append-Only File&#xf…...

FPGA面试问题整理

1. 逻辑设计中竞争与冒险概念&#xff0c;如何识别和消除&#xff1f; 竞争&#xff1a;在组合逻辑电路中&#xff0c;信号经过多条路径到达输出端&#xff0c;每条路径经过的逻辑门不同存在时差&#xff0c;在信号变化的瞬间存在先后顺序。这种现象叫竞争。 冒险&#xff1a;由…...

3Done学习笔记

一、基本操作 1、旋转视角 使用左下角立方体选择&#xff1b; 右键可以拖动视角&#xff1b; 中间滑轮按住拖动整个舞台界面。 2、平移和旋转 右键选择移动&#xff0c;有两种方式。 第一种选择起始点&#xff0c;按照起始点位置移动到终止点&#xff08;边、角、中心点…...

AI学习指南深度学习篇-卷积层详解

AI学习指南深度学习篇-卷积层详解 一、引言 随着人工智能技术的不断发展&#xff0c;深度学习作为人工智能领域的热门分支之一&#xff0c;正在逐渐成为各个领域的核心技术。而在深度学习中&#xff0c;卷积神经网络&#xff08;Convolutional Neural Network&#xff0c;CNN…...

2024年TI杯E题-三子棋游戏装置方案分享-jdk123团队-第二弹 手搓机械臂

第一弹赛题的选择与前期方案的准备 opencv调用摄像头bug的解决 机械臂的组装 采用三个舵机&#xff0c;组成一个三自由度的机械臂。 并且利用电磁吸盘的方式&#xff0c;完成对棋子的抓取工作&#xff0c;后面的事实证明&#xff0c;在预算不足的情况下&#xff0c;队友手搓…...

如何在Java、C、Ruby语言中使用Newscatcher API

Newscatcher 世界实时新闻聚合API 一款强大的数据服务工具&#xff0c;它通过先进的网络爬虫技术&#xff0c;实时从全球超过70,000个新闻源聚合新闻内容。这个API能够提供全面、多角度的新闻报道&#xff0c;包括但不限于标题、作者、发布日期、全文内容以及媒体资源链接。它使…...

集合: Collection的成员方法和相关实现类

Collection: - List(有序【指的是存储和取出的顺序是一致的】且可以发生重复&#xff0c;且有索引的概念) - ArrayList&#xff1a; 底层数据结构是数组&#xff0c;查询快&#xff0c;增删慢&#xff0c;线程不安全的&#xff0c;效率高。 - …...

过滤器与监听器:深入了解 Java Web 开发中的核心概念

在 Java Web 开发中&#xff0c;过滤器&#xff08;Filter&#xff09;和监听器&#xff08;Listener&#xff09;是两个重要的组件&#xff0c;它们帮助开发者在请求处理的各个阶段进行预处理和后处理。这篇博客将深入探讨这两个概念&#xff0c;并展示它们如何在实际应用中发…...

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…...

装饰模式(Decorator Pattern)重构java邮件发奖系统实战

前言 现在我们有个如下的需求&#xff0c;设计一个邮件发奖的小系统&#xff0c; 需求 1.数据验证 → 2. 敏感信息加密 → 3. 日志记录 → 4. 实际发送邮件 装饰器模式&#xff08;Decorator Pattern&#xff09;允许向一个现有的对象添加新的功能&#xff0c;同时又不改变其…...

React Native在HarmonyOS 5.0阅读类应用开发中的实践

一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强&#xff0c;React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 &#xff08;1&#xff09;使用React Native…...

多种风格导航菜单 HTML 实现(附源码)

下面我将为您展示 6 种不同风格的导航菜单实现&#xff0c;每种都包含完整 HTML、CSS 和 JavaScript 代码。 1. 简约水平导航栏 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport&qu…...

Java求职者面试指南:计算机基础与源码原理深度解析

Java求职者面试指南&#xff1a;计算机基础与源码原理深度解析 第一轮提问&#xff1a;基础概念问题 1. 请解释什么是进程和线程的区别&#xff1f; 面试官&#xff1a;进程是程序的一次执行过程&#xff0c;是系统进行资源分配和调度的基本单位&#xff1b;而线程是进程中的…...

RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)

RabbitMQ 一、RabbitMQ概述 RabbitMQ RabbitMQ最初由LShift和CohesiveFT于2007年开发&#xff0c;后来由Pivotal Software Inc.&#xff08;现为VMware子公司&#xff09;接管。RabbitMQ 是一个开源的消息代理和队列服务器&#xff0c;用 Erlang 语言编写。广泛应用于各种分布…...

STM32HAL库USART源代码解析及应用

STM32HAL库USART源代码解析 前言STM32CubeIDE配置串口USART和UART的选择使用模式参数设置GPIO配置DMA配置中断配置硬件流控制使能生成代码解析和使用方法串口初始化__UART_HandleTypeDef结构体浅析HAL库代码实际使用方法使用轮询方式发送使用轮询方式接收使用中断方式发送使用中…...

上位机开发过程中的设计模式体会(1):工厂方法模式、单例模式和生成器模式

简介 在我的 QT/C 开发工作中&#xff0c;合理运用设计模式极大地提高了代码的可维护性和可扩展性。本文将分享我在实际项目中应用的三种创造型模式&#xff1a;工厂方法模式、单例模式和生成器模式。 1. 工厂模式 (Factory Pattern) 应用场景 在我的 QT 项目中曾经有一个需…...

书籍“之“字形打印矩阵(8)0609

题目 给定一个矩阵matrix&#xff0c;按照"之"字形的方式打印这个矩阵&#xff0c;例如&#xff1a; 1 2 3 4 5 6 7 8 9 10 11 12 ”之“字形打印的结果为&#xff1a;1&#xff0c;…...

Python环境安装与虚拟环境配置详解

本文档旨在为Python开发者提供一站式的环境安装与虚拟环境配置指南&#xff0c;适用于Windows、macOS和Linux系统。无论你是初学者还是有经验的开发者&#xff0c;都能在此找到适合自己的环境搭建方法和常见问题的解决方案。 快速开始 一分钟快速安装与虚拟环境配置 # macOS/…...