VS2019连接MySQL

VS2019连接MySQL

  • 下载MySQL Connector/C++
  • 配置头文件,库文件路径
    • 配置头文件路径
    • 配置库的路径
    • 复制dll文件
  • MySQL的用户设置
    • 将权限赋值给新用户
  • 编写代码
  • 往数据库写入

老师布置的作业让我们用VS2019连接MySQL实现一个小型的日志系统,中间踩了很多的坑,写篇博客记录一下:

下载MySQL Connector/C++

首先看看自己的MySQL文件夹下,有没有这个文件,如果是默认路径,一般会放到:Program Files下:
在这里插入图片描述
点进去:

如果有这个,说明在装MySQL时已经将Connector/C++ 装好了,此时可以直接进入下一步,如果没有的话,可以访问官网下载:

https://dev.mysql.com/downloads/connector/cpp/
在这里插入图片描述

配置头文件,库文件路径

我们装的库是一个第三方库,所以VS不能自动识别,我们得自己配置一下。
新创建一个项目:
在这里插入图片描述此时第一步把x86换成x64,我们的这个Connector/C++是64位的库,如果不换,之后会有一堆奇怪的错误:
在这里插入图片描述
添加一个头文件和源文件:
在这里插入图片描述
在这里插入图片描述

配置头文件路径

此时,鼠标右击这里:
在这里插入图片描述选择属性:

在这里插入图片描述选择C/C++下面的常规(如果没有这个选项,先添加一个源文件就行了)
在这里插入图片描述编辑附加包含目录
在这里插入图片描述在这里插入图片描述

把Connector/C++ 下的include路径添加进去:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
还要包括jbdc的路径
在这里插入图片描述在这里插入图片描述

配置库的路径

换到下面的连接器的常规选项:
在这里插入图片描述
编辑附加库目录:将lib64下的vs14路径写进去:
在这里插入图片描述在这里插入图片描述
然后切到输入选项:
在这里插入图片描述编辑附加依赖项:将这两个库的名字粘贴进去,不带路径

在这里插入图片描述在这里插入图片描述不要网络点右下角的应用:
在这里插入图片描述

复制dll文件

最后,需要将lib64下的dll文件复制到C:\Windows\System32或者是复制到项目的目录中:

在这里插入图片描述在这里插入图片描述或者这里:
在这里插入图片描述
现在我们包几个头文件试试:

#pragma once
#include <mysql_driver.h>  
#include <mysql_connection.h>  
#include <cppconn/statement.h>  
#include <cppconn/resultset.h>  
#include <cppconn/prepared_statement.h>

在这里插入图片描述
如果没有报错,说明配置成功。

MySQL的用户设置

我们配置好了VS,现在我们来配置MySQL的用户,这里我创建一个新用户来演示,首先,我们得以以root用户身份登录到MySQL服务器。打开命令行界面

mysql -u root -p

在这里插入图片描述
以root的身份创建一个新用户:假设你要创建一个名为newuser的用户,并且该用户只能从localhost连接,你可以执行以下命令:

CREATE USER 'newuser'@'localhost' IDENTIFIED BY 'password';

请将newuser替换为你想要创建的用户名,将localhost替换为允许该用户连接的主机名或IP地址,将password替换为用户的密码。

在这里插入图片描述

将权限赋值给新用户

MySQL提供了多种权限级别,可以针对特定数据库、特定表甚至特定操作来分配权限。例如,如果你想给 ‘new_user’ 用户在名为 ‘my_database’ 的数据库中所有表的所有权限,可以使用以下命令:

GRANT ALL PRIVILEGES ON my_database.* TO 'new_user'@'localhost';

这里的 ALL PRIVILEGES 表示赋予所有权限,包括但不限于查询、插入、更新、删除、创建表等。ON my_database.* 指明了权限适用的数据库及其所有表。

假设我这里有一个my_log的数据库,那我应该这么写:
在这里插入图片描述在完成权限分配后,通常需要执行 FLUSH PRIVILEGES; 命令来刷新MySQL的权限缓存,使其立即生效:

FLUSH PRIVILEGES;

在这里插入图片描述
此时MySQL这边的配置也搞定。

编写代码

#pragma once
#include <mysql_driver.h>  
#include <mysql_connection.h>  
#include <cppconn/statement.h>  
#include <cppconn/resultset.h>  
#include <cppconn/prepared_statement.h>class Logger
{
public:Logger(){}private:std::string dbName = "my_log";std::string dbUser = "luoshui";std::string dbPassword = "xiangzihao137";std::string dbHost = "localhost";std::string dbPort = "3306";sql::mysql::MySQL_Driver* driver;sql::Connection* con;
};

这里出现了两个新的东西,我们来解释一下:

在C++中,特别是在使用MySQL Connector/C++与MySQL数据库交互时,你提到的这两行代码是关于对象指针的声明。让我们详细解释一下它们的意思:
sql::mysql::MySQL_Driver* driver
sql::mysql::MySQL_Driver:这是MySQL Connector/C++库中的一个类,它表示一个MySQL数据库驱动。这个驱动是连接到MySQL服务器所必需的。
driver;:这表示声明了一个指向sql::mysql::MySQL_Driver类对象的指针,名为driver。在实际使用之前,你通常会初始化这个指针,让它指向一个MySQL_Driver对象。
sql::Connection* con;
sql::Connection:这是MySQL Connector/C++库中的另一个类,它表示一个到MySQL服务器的连接。通过这个连接,你可以执行SQL查询、获取结果等。
* con;:这表示声明了一个指向sql::Connection类对象的指针,名为con。在实际使用之前,你通常会使用这个指针来创建一个到MySQL服务器的连接。
为了使用这些指针,你通常会按照以下步骤操作:
初始化driver指针,让它指向一个新的MySQL_Driver对象。
使用driver指针来创建与MySQL服务器的连接,并将返回的连接对象赋值给con指针。
使用con指针来执行各种数据库操作,如执行查询、获取结果等。

这些操作可能会涉及到异常处理,因为数据库连接和操作可能会因为各种原因(如连接失败、SQL错误等)而失败。因此,在使用这些类时,通常需要适当的错误检查和异常处理。

#pragma once
#include <mysql_driver.h>  
#include<sstream>
#include <mysql_connection.h>  
#include <cppconn/statement.h>  
#include <cppconn/resultset.h>  
#include <cppconn/prepared_statement.h>class Logger
{
public:Logger(){try{//实例化driver对象driver = sql::mysql::get_mysql_driver_instance();//使用字符流,完成连接地址std::stringstream url;url << "tcp://" << dbHost << ":" << dbPort << "/" << dbName;//开始连接con = driver->connect(url.str().c_str(), dbUser.c_str(), dbPassword.c_str());}catch(sql::SQLException& e){//如果失败,捕捉异常std::cerr << e.what();}}~Logger() {delete con;}private:std::string dbName = "my_log";std::string dbUser = "luoshui";std::string dbPassword = "xiangzihao137";std::string dbHost = "localhost";std::string dbPort = "3306";sql::mysql::MySQL_Driver* driver;sql::Connection* con;
};

此时我们用DataGrip:点击+
在这里插入图片描述添加MySQL的连接:
在这里插入图片描述
创建新的连接:
在这里插入图片描述在这里插入图片描述
此时创建好了链接,我们创建一个数据库(就是我们之前假设的那个my_log):
在这里插入图片描述在这里插入图片描述此时一输入完名字,DataGrip会自动识别我们之前为这个my_log设置的权限。

这里可能DataGrip会报错,但是不用管,点击取消发现我们的my_log已经创建好了:
在这里插入图片描述

往数据库写入

这时候就可以准备往数据库写入了:

   //往数据库中写入数据void logToDatabase(const std::string& message) {try{sql::PreparedStatement* pstmt = con->prepareStatement("INSERT INTO logs (message) VALUES (?)");pstmt->setString(1, message);pstmt->executeUpdate();pstmt->close(); // 或者使用delete pstmt; (取决于MySQL Connector/C++的版本)}catch (sql::SQLException& e) {std::cerr << "# ERR: " << e.what();}}

这段代码是使用MySQL Connector/C++库编写的一个Java风格的C++代码片段,用于向名为logs的数据库表中插入一行记录。下面逐行解释:

  1. sql::PreparedStatement* pstmt = con->prepareStatement("INSERT INTO logs (message) VALUES (?)");
  • 创建一个PreparedStatement对象,这个对象预先准备了一条SQL插入语句。在SQL语句中,message列使用问号(?)作为占位符,这是预处理语句的标准做法,可以防止SQL注入攻击,同时也允许我们在执行时动态地插入值。
  1. pstmt->setString(1, message);
  • 为预处理语句的第一个参数(对应SQL语句中的第一个问号)设置值。这里传入的是一个字符串message。数字1表示参数的位置,从1开始计数。
  1. pstmt->executeUpdate();
  • 执行预处理好的SQL语句,也就是执行插入操作,将前面设置好的字符串message插入到logs表的message列中。
  1. pstmt->close();
  • 关闭PreparedStatement对象,释放相关资源。在某些MySQL Connector/C++版本中,可能需要手动关闭这个对象以确保资源得到妥善管理。在现代版本的连接器中,通常推荐使用智能指针或其他自动资源管理机制来自动关闭和释放资源,但如果使用原始指针,则需要手动关闭。
  • 注意:有时根据具体的库实现和内存管理策略,可能需要使用delete pstmt;来删除对象,释放内存,但多数现代的C++数据库驱动程序倾向于使用RAII(Resource Acquisition Is Initialization)原则,即通过构造函数获取资源并在析构函数中自动释放资源,因此直接调用close()方法即可。

总而言之,这段代码是为插入日志记录到数据库做准备、设置参数并执行插入操作,最后清理资源。

注意这里:sql::PreparedStatement* pstmt = con->prepareStatement("INSERT INTO logs (message) VALUES (?)")这里的意思是:想在数据库my_log下的logs表中的message插入数据,如果这张表还没有的话,我们得提前创建一下:

CREATE TABLE IF NOT EXISTS my_log.logs (id INT AUTO_INCREMENT PRIMARY KEY,message TEXT NOT NULL,timestamp TIMESTAMP DEFAULT CURRENT_TIMESTAMP);

在这里插入图片描述

#define _CRT_SECURE_NO_WARNINGS 1
#include"log.hpp"int main()
{Logger logge;logge.logToDatabase("This is a massage");return 0;
}

但是这个时候编的过,但是运行不过:
在这里插入图片描述
这个是因为内存碎片过多,目前没有什么好的解决办法,我们可以将Debug换成Release模式:

在这里插入图片描述这个时候会疯狂报红,是因为我们之前的设置是在Debug下,换成Release就的重新从头再把头文件库包括一遍。

在这里插入图片描述重新配置之后,爆红就消失了。
再次运行:
在这里插入图片描述返回我们的表中,看看么message这一项:
在这里插入图片描述发现已经被成功写入了,我们可以换一下:
在这里插入图片描述
在这里插入图片描述

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

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

相关文章

RecyclerView 调用 notifyItemInserted 自动滚动到底部的问题

项目中发现一个奇怪的现象 RecyclerView 加载完数据以后&#xff0c;调用 notifyItemInserted 方法&#xff0c;RecyclerView 会滑动到底部。 简化后的效果图&#xff1a; 因为这个 RecyclerView 的适配器有一个 FootViewHolder&#xff0c;所以怀疑是 FootViewHolder 的问题…

MTU/TCPMSS/VLAN/ACCESS/TRUNK/HYBRID

MTU RFC标准定义以太网的默认MTU值为1500 最小64字节是为了保证最极端的冲突能被检测到&#xff0c;64字节是能被检测到的最小值&#xff1b;最大不超过1518字节是为了防止过长的帧传输时间过长而占用共享链路太长时间导致其他业务阻塞。所以规定以太网帧大小为64~1518字节&am…

java文件File和IO流(二)-- IO流,递归,数据流,打印流,转换流等等

IO流 IO流之数据流 在IO流中&#xff0c;可以通过DataInputStream和DataOutputStream字节流直接操作基本数据类型和字符串 DataOutputStream import java.io.*;//TODO 数据流&#xff0c;简单的说&#xff0c;就是容许字节流直接操作基本数据类型和字符串。 public class D…

Golang-Gorm-快速上手

Gorm文档 GORM文档地址 安装依赖 go get -u "gorm.io/driver/mysql"go get -u "gorm.io/gorm"连接数据库 默认连接方式 func main() {// 参考 https://github.com/go-sql-driver/mysql#dsn-data-source-name 获取详情dsn : "user:passtcp(127.0.0…

【C++第五课-C/C++内存管理】C/C++的内存分布、new/delete、new和delete的实现原理

目录 C/C的内存分布new/deletenew内置类型使用new自定义类型使用newnew失败 delete内置类型使用delete自定义类型使用delete new和delete的实现原理new[] 和delete[]的补充知识 定位new&#xff08;了解&#xff09;常见面试题 C/C的内存分布 频繁的new/delete堆容易产生内存碎…

算法学习——LeetCode力扣动态规划篇10(583. 两个字符串的删除操作、72. 编辑距离、647. 回文子串、516. 最长回文子序列)

算法学习——LeetCode力扣动态规划篇10 583. 两个字符串的删除操作 583. 两个字符串的删除操作 - 力扣&#xff08;LeetCode&#xff09; 描述 给定两个单词 word1 和 word2 &#xff0c;返回使得 word1 和 word2 相同所需的最小步数。 每步 可以删除任意一个字符串中的一个…

2023年第十四届蓝桥杯大赛软件类省赛C/C++研究生组真题(代码完整题解)

C题-翻转⭐ 标签:贪心 简述:如果 S 中存在子串 101 或者 010,就可以将其分别变为 111 和 000,操作可以无限重复。最少翻转多少次可以把 S 变成和 T 一样。 链接: 翻转 思路:要求步骤最少->S每个位置最多修改一次->从头开始遍历不匹配就翻转->翻转不了就-1 …

汇编语言——用INT 21H 的A号功能,输入一个字符串存放在内存,倒序输出

用INT 21H 的A号功能&#xff0c;输入一个字符串“Hello, world!”&#xff0c;存放在内存&#xff0c;然 后倒序输出。 在DOS中断中&#xff0c;INT 21H是一个常用的系统功能调用中断&#xff0c;它提供了多种功能&#xff0c;其中A号功能用于字符串的输入。 在使用这个功能时…

康耐视visionpro-CogAcqFifoTool工具详细说明

CogAcqFifoTool操作说明&#xff1a; ① 打开工具栏&#xff0c;双击或点击鼠标拖拽 添加CogAcqFifoTool ②.从图片采集设备/图像采集卡列表里选择对应的相机&#xff0c;视频格式选择图像格式。 Mono表示黑白图像&#xff0c;RGB表示彩色相机。点击初始化取相初始化相机。 ③…

灵动翻译音频文件字幕提取及翻译;剪映视频添加字幕

参考&#xff1a;视频音频下载工具 https://tuberipper.com/21/save/mp3 1、灵动翻译音频文件字幕提取及翻译 灵动翻译可以直接chorme浏览器插件安装&#xff1a; 点击使用&#xff0c;可以上传音频文件 上传后自动翻译&#xff0c;然后点击译文即可翻译成中文&#xff0c;…

排序——非基于比较的排序

本专栏和大家分享关于排序的算法,其中有插入排&#xff08;直接插入排序和希尔排序&#xff09;、选择排序&#xff08;直接选择排序和堆排&#xff09;、交换排序&#xff08;冒泡排序和快速排序&#xff09;、归并排序以及其他非基于比较的排序 本文与大家分享非基于比较的排…

MySQL学习之连接查询

笛卡尔乘积现象 在表的连接查询方面有一种现象被称为&#xff1a;笛卡尔积现象。 笛卡尔积现象&#xff1a; 当两张表进行连接查询的时候&#xff0c;没有任何条件进行限制&#xff0c;最终的查询结果条数是两张表记录条数的乘积。 select ename,dname from emp,dept; 避免…

算法系列--动态规划--特殊的状态表示--分析重复子问题

&#x1f495;"轻舟已过万重山!"&#x1f495; 作者&#xff1a;Lvzi 文章主要内容&#xff1a;算法系列–算法系列–动态规划–特殊的状态表示–分析重复子问题 大家好,今天为大家带来的是算法系列--动态规划--特殊的状态表示--分析重复子问题 一.组合总数IV 链接…

实现富文本的三部曲

1、引入 ueditor.config.js ueditor.all.min.js lang/zh-cn/zh-cn.js 2、编辑器显示处 id"content" <textarea id"content" name"content"></textarea> 3、底部 <script type"text/javascript"> //实例化编辑器 …

搜索与图论——Floyd算法求最短路

floyd算法用来求多源汇最短路 用邻接矩阵来存所有的边 时间复杂度O(n^3) #include<iostream> #include<cstring> #include<algorithm>using namespace std;const int N 20010,INF 1e9;int n,m,k; int g[N][N];void floyd(){for(int k 1;k < n;k ){f…

网络基础(二)——序列化与反序列化

目录 1、应用层 2、再谈“协议” 3、网络版计算器 Socket.hpp TcpServer.hpp ServerCal.hpp ServerCal.cc Protocol.hpp ClientCal.cc Log.hpp Makefile 1、应用层 我们程序员写的一个个解决我们实际问题&#xff0c;满足我们日常需求的网络程序&#xff0c;都是在…

QT+Opencv+yolov5实现监测

功能说明&#xff1a;使用QTOpencvyolov5实现监测 仓库链接&#xff1a;https://gitee.com/wangyoujie11/qt_yolov5.git git本仓库到本地 一、环境配置 1.opencv配置 将OpenCV-MinGW-Build-OpenCV-4.5.2-x64文件夹放在自己的一个目录下&#xff0c;如我的路径&#xff1a; …

appium辅助自动化工具-- Appium studio

这里我要给大家介绍一款appium辅助自动化测试工具appium studio&#xff0c;你没看错&#xff0c;不是android studio&#xff0c;也不是appium android studio&#xff0c;就是appium studio&#xff01; 下载地址&#xff1a; Appium Studio | Digital.ai Continuous Test…

智慧公厕的技术融合策略

智慧公厕是迎合现代城市发展需要的一项重要基础设施&#xff0c;其设计的技术融合策略在实现公共厕所泛在感知、互通互联、协同构筑智慧城市等方面起到了关键作用。本文将以智慧公厕源头实力厂家广州中期科技有限公司&#xff0c;大量精品案例现场实景实图实例&#xff0c;从物…

Makefile:动态库的编译链接与使用(六)

1、动态链接库 动态链接库&#xff1a;不会把代码编译到二进制文件中&#xff0c;而是运行时才去加载&#xff0c;所以只需要维护一个地址 动态&#xff1a;运行时才去加载&#xff0c;即所谓的动态加载连接&#xff1a;指库文件和二进制程序分离&#xff0c;用某种特殊的手段…