VS2019连接MySQL

news2025/1/10 11:52:58

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.coloradmin.cn/o/1559198.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

神经网络与深度学习(一)误差反传BP算法

误差反传BP算法 1多层感知机1.1XOR问题1.2多层感知机 2.BP算法2.1简述2.2详解2.2.1输入输出模型2.2.2梯度下降算法迭代2.2.3前向传播在输出端计算误差2.2.4误差反传--输出层2.2.5误差反传--隐含层2.2.6误差反传--总结 1多层感知机 1.1XOR问题 线性不可分问题&#xff1a; 无法…

C++心决之命名空间、重载函数和引用

目录 1. C关键字(C98) 2. 命名空间 2.1 命名空间定义 2.2 命名空间使用 3. C输入&输出 4. 缺省参数 4.1 缺省参数概念 4.2 缺省参数分类 5. 函数重载 5.1 函数重载概念 5.2 C支持函数重载的原理--名字修饰(name Mangling) 6. 引用 6.1 引用概念 6.2 引用特性…

Java多线程:定位死锁

检测死锁可以使用jconsole工具&#xff0c;或使用jps定位进程id&#xff0c;再用jstack定位死锁 方案1&#xff1a; 1. 先用jps查看所有的java进程id 2. jstack 进程id定位死锁 3. 查看死锁结果 方案2:从jdk的安装路径中找到bin目录, 点击jconsole

Linux线程原理以及基础操控函数使用(1)

线程原理 这一个课程的笔记 LWP&#xff1a;light weight process 轻量级的进程&#xff0c;本质仍是进程(在Linux环境下) 进程&#xff1a;独立地址空间&#xff0c;拥有PCB 线程&#xff1a;有独立的PCB&#xff0c;但没有独立的地址空间(共享) 区别&#xff1a;在于是否共…

算法6.4-6.6DFS

一个不知名大学生&#xff0c;江湖人称菜狗 original author: Jacky Li Email : 3435673055qq.com Time of completion&#xff1a;2024.03.27 Last edited: 2024.03.27 目录 算法6.4-6.6DFS 第1关&#xff1a;算法6.5采用邻接矩阵表示图的深搜 任务描述 相关知识 编程要求…

数据结构——优先级队列及多服务台模拟系统的实现

一、优先级队列的定义和存储 优先级队列定义&#xff1a;优先级高的元素在队头&#xff0c;优先级低的元素在队尾 基于普通线性表实现优先级队列&#xff0c;入队和出队中必有一个时间复杂度O(n),基于二叉树结构实现优先级队列&#xff0c;能够让入队和出队时间复杂度都为O(log…

RabbitMQ--04--发布订阅模式 (fanout)-案例

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 发布订阅模式 (fanout)---案例前言RabbitListener和RabbitHandler的使用 1.通过Spring官网快速创建一个RabbitMQ的生产者项目2.导入项目后在application.yml文件中配…

Python编程-并发编程基础梳理与高级特性案例讲解

Python编程-并发编程基础梳理与高级特性案例讲解 同步、异步通信与IO阻塞 同步&#xff08;Synchronous&#xff09;和异步&#xff08;Asynchronous&#xff09;通信是两种不同的通信方式&#xff0c;涉及到处理任务的时序和相互等待的关系。同步通信简单直观&#xff0c;但可…

Unity 使用TrailRenderer制作拖尾效果

使用TrailRenderer实现拖尾效果&#xff0c;具体操作步骤如下&#xff1a; 1、创建一个空对象 在Unity场景中创建一个空对象 2、添加TrailRenderer组件 选择步骤1创建的空对象&#xff0c;然后在Inspector面板中点击“Add Component”按钮&#xff0c;搜索并添加TrailRende…

蓝桥杯第七届大学B组详解

目录 1.煤球数量&#xff1b; 2.生日蜡烛&#xff1b; 3.凑算式 4.方格填数 5.四平方和 6.交换瓶子 7.最大比例 1.煤球数量 题目解析&#xff1a;可以根据题目的意思&#xff0c;找到规律。 1 *- 1个 2 *** 3个 3 ****** 6个 4 ********** 10个 不难发现 第…

C++入门知识详细讲解

C入门知识详细讲解 1. C简介1.1 什么是C1.2 C的发展史1.3. C的重要性1.3.1 语言的使用广泛度1.3.2 在工作领域 2. C基本语法知识2.1. C关键字(C98)2.2. 命名空间2.2 命名空间使用2.2 命名空间使用 2.3. C输入&输出2.4. 缺省参数2.4.1 缺省参数概念2.4.2 缺省参数分类 2.5. …

Linux - 第三节

改变用户类型 su 仅单纯的进行身份变化 依旧处于普通用户里面 su - 进行重新登录更改身份 退出用exit / ctrld su 用户名 改成成其他身份 对一条命令进行提权 sudo command r:可读 w:可写 x:可执行 -:对应的权限位置&#xff0c;没有权限 去掉所有权限 chmod u…

Qt中继承QCheckBox的类结合QTableWidget实现多选并且每个多选的id都不一样

1.相关描述 继承QCheckBox的类MyCheckBox&#xff0c;利用QTableWidget的setCellWidget方式添加MyCheckBox类的对象 2.相关页面 3.相关代码 mycheckbox.h #ifndef MYCHECKBOX_H #define MYCHECKBOX_H#include <QCheckBox> #include <QObject>class MyCheckBox : pu…

Elvis Presley 英文阅读

Elvis Presley 官方翻译 One of the most popular American singers of thetwentieth century was Elvis Presley. He made theRock & Roll music popular around the world. He sold millions of records and made manysuccessful films, and he helped change the dir…

基于SSM医院病历管理系统

基于SSM医院病历管理系统的设计与实现 摘要 病历管理系统是医院管理系统的重要组成,在计算机技术快速发展之前&#xff0c;病人或者医生如果想记录并查看自己的健康信息是非常麻烦的&#xff0c;因为在以往病人的健康信息通常只保存在自己的病历卡或者就诊报告中&#xff0c;…

TikTok直播专线:解决出海网络问题痛点,提升商业效率

近年来&#xff0c;TikTok作为全球最受欢迎的社交媒体平台之一&#xff0c;成为商家获取商机与市场的重要平台。然而&#xff0c;尽管商家纷纷进入TikTok&#xff0c;试图借助其强大的社交网络进行产品推广和销售&#xff0c;但在TikTok平台进行直播时&#xff0c;往往会遇到网…

何时应用 RAG 与微调

充分发挥 LLM 的潜力需要在检索增强生成&#xff08;RAG&#xff09;和微调之间选择正确的技术。 让我们来看看何时对 LLM、较小的模型和预训练模型使用 RAG 与微调。我们将介绍&#xff1a; LLM 和 RAG 的简要背景RAG 相对于微调 LLM 的优势何时针对不同模型大小对 RAG 进行…

WebGIS 之 vue3+vite+ceisum

1.项目搭建node版本在16以上 1.1创建项目 npm create vite 项目名 1.2选择框架 vuejavaScript 1.3进入项目安装依赖 cd 项目名 npm install 1.4安装cesium依赖 pnpm i cesium vite-plugin-cesium 1.5修改vite.config.js文件 import { defineConfig } from vite import vue fr…

labelme AI 模型运用

一、lebelme 1、界面介绍 点击上图位置&#xff0c;选择对应的模型。这里我每个模型都测试了一下&#xff0c;EfficientSam这个模型最好用&#xff0c;准确率和速度都ok。 2、使用方法 目标框标注方法&#xff1a;点左上角【编辑】-> 【Create Ai-Mask】就可以标志了&…

在 Three.js 中,OBJExporter 是一个用于将 Three.js 中的场景导出为 OBJ 格式的类。

demo案例 在 Three.js 中&#xff0c;OBJExporter 是一个用于将 Three.js 中的场景导出为 OBJ 格式的类。下面是关于它的入参、出参、属性和方法的解释&#xff1a; 类名&#xff1a;OBJExporter 构造函数&#xff1a; THREE.OBJExporter()说明&#xff1a; 创建一个 OBJE…