QT C++实践|超详细数据库的连接和增删改查操作|附源码
0:前言
🪧 什么情况需要数据库?
- 1 大规模的数据需要处理(比如上千上万的数据量)
- 2 需要把数据信息存储起来,无论是本地还是服务上,而不是断电后数据信息就消失了。
如果不是上面的原因化,一般可以使用数组来处理。
🪧一般常使用的数据库驱动是MYSQL和QSQLITE。二者区别在于,前者用于服务器存储信息,后者用于本地存储信息。并且QSQLITE主要用于嵌入式,占用资源非常低,占用内存小,通常几百k就搞定。’
这里博主因为对
MySQL熟悉一些,就使用MySQL来进行数据库的连接
一、Mysql的安装
因为我们项目的方案是程序的运行以及相关数据的存储都在一台主机上,所以不论打不打包。首先要在主机上安装Mysql的。Mysql的安装教程我参考的是这个:MySQL安装和配置教程(超详细版本)
安装好后,利用命令行或其他工具在MySQL中创建一个存储项目数据的数据库,方便之后使用QT用代码对数据库进行连接、建表和增删改查的操作:
Win+r打开cmd:输入命令“mysql -u root -p”,按下回车键- 输入MySQL密码,按下回车键
- 创建数据库
create database [if not exists] 数据库名 [default charset 字符集] [collate 排序规则];
注意:不要忘记末尾的分号
二、通过ODBC连接MySQL数据库
官方解释:
ODBC(Open Database Connectivity,开放数据库互连)提供了一种标准的API(应用程序编程接口)方法来访问数据库管理系统(DBMS)。这些API利用SQL来完成其大部分任务。ODBC本身也提供了对SQL语言的支持,用户可以直接将SQL语句送给ODBC。ODBC的设计者们努力使它具有最大的独立性和开放性:与具体的编程语言无关,与具体的数据库系统无关,与具体的操作系统无关。
简单的说就是我的qt中含有ODBC的驱动:

所以利用ODBC去使用MySQL的数据库
2.1:下载ODBC
官网

选择和qt编译器相同的字节比如我的qt使用的是64字节的
下载完成后,点卡下载下来的.msi文件并运行:

下载一直点击next:

选择custom,表示自定义安装(以便修改安装位置)


最后一直点击next然后再install即可,等待安装好后即可。
安装好之后,关闭窗口,搜索ODBC,并运行程序:

2.2:使用ODBC连接MySQL数据库
添加MySQL的DSN

红色的可以随便填,是自己对于ODBC驱动的描述,粉色的是MySQL的用户名和密码,数据库选择你在MySQL中创建的数据库名字即可:

点击Test,测试是否能够成功连接;如果出现下图说明连接成功

然后点击应用和确定进行添加:

2.3:qt通过ODBC连接MySQL
现在main.cpp中加入下面代码测试一下能不能在qt中连接成功:
QSqlDatabase db = QSqlDatabase::addDatabase("QODBC");db.setHostName("127.0.0.1");db.setPort(3306);db.setDatabaseName("是你在ODBC中创建的Data source name");db.setUserName("用户名");db.setPassword("密码");bool ok = db.open();if (ok){QMessageBox::information(this, "infor", "success");}else {QMessageBox::information(this, "infor", "open failed");qDebug()<<"error open database because"<<db.lastError().text();}
Tips:这里可能会报错:VS2019 C1083 无法打开包括文件: “QSqlDatabase”
是vs里面项目配置的问题。
看到这个错误应该是没有dll,然后sql很容易想到数据库,所以在qt模块里添加,比较方便。


不出意外运行程序可以成功连接:

之后在主机进行安装时,除了打包程序,应该还需要安装对应的Mysql和ODBC
2.4:qt通过ODBC操作数据库
连接上数据库一般就是使用数据库,进行对数据库的增删改查。
这里有三种方法。
🪧 首先单独建立一个头文件来处理数据库连接,如建立头文件connection.h:
#ifndef CONNECTION_H
#define CONNECTION_H#include <QMessageBox>
#include <QSqlDatabase>
#include <QSqlQuery>
#include <QDebug>static bool createConnection(){//连接第一个数据库//QMYSQLQSqlDatabase db = QSqlDatabase::addDatabase("QMYSQL", "connection1");//需要使用的数据库驱动和联检建立的名称(方便建立多个数据库连接【使用不同的数据库时】区分)db.setHostName("127.0.0.1");//连接地址db.setUserName("root");//数据库账户db.setPassword("root");//密码db.setPort(8889);//端口//test_majiang.dbdb.setDatabaseName("test_majiang");//需要用到的数据库if (!db.open()) {//如果数据库连接失败,则弹出//critical(QWidget *parent, const QString &title,//const QString &text,//QMessageBox::StandardButtons buttons = Ok,//QMessageBox::StandardButton defaultButton = NoButton)QMessageBox::critical(0, "Cannot open database","Unable to establish a database connection", QMessageBox::Cancel);return false;}return true;
}#endif // CONNECTION_H
如果需要移除一个数据库连接,可以使用:
QSqlDatabase::close();//关闭数据库QSqlDatabase::removeDatabase();//移除该连接
2.4.1:使用QSqlQuery
头文件
#include "connection.h"
#include <QVariant>
#include <QSqlDriver>
#include <QSqlRecord>
#include <QSqlField>
#include <QSqlError>
- 使用数据库前的准备:
//创建数据库连接if(!createConnection()) return 1;//返回情况可以替换,视不同情况而定//指定某个数据库连接QSqlDatabase db2 = QSqlDatabase::database("connection1");
- 开始对数据进行操作:
首先创建QSqlQuery 对象,然后进行操作。
QSqlDatabase db2 = QSqlDatabase::database("connection2");QSqlQuery query2(db2);
- 进行创表和插入值:
// qDebug() << "connection2:";//创建表,并插入值query2.exec("create table student (id int primary key,""name varchar(20))");query2.exec("insert into student values(0, 'Mike')");query2.exec("insert into student values(1, 'Lili')");query2.exec("insert into student values(2, 'Jame')");
- 批量处理
上面的单条插入语句明显比较麻烦,可以使用批量插入数据:
query2.exec("insert into student(id,name) values(3,'Qinsong')");//单挑操作//名称绑定query2.prepare("insert into student(id, name) values(:id, :name)");int idValue = 4;QString nameValue = "Songjiang";query2.bindValue(":id", idValue);//绑定数据query2.bindValue(":name", nameValue);query2.exec();//执行//位置绑定query2.prepare("insert into student(id, name) values(?, ?)");int idValue2 = 5;QString nameValue2 = "LingChong";query2.addBindValue(idValue2);//绑定数据query2.addBindValue(nameValue2);if(!query2.execBatch()) qDebug() <<"位置绑定:" <<query2.lastError() <<endl;//如果执行不成功执行//批量处理query2.prepare("insert into student(id, name) values(?,?)");QVariantList ids;ids << 6 << 7 << 8;query2.addBindValue(ids);QVariantList names;names << "Qinghua" << "Nanda" << "Zhongkeda";query2.addBindValue(names);if(!query2.execBatch()) qDebug() << query2.lastError()<< endl;
- 进行查询并输出查询结果:
query2.exec("select * from student");//执行sql语句while(query2.next()){qDebug()<< query2.value(0).toInt() << query2.value(1).toString();}
- 查看数据驱动支持特性
查看当前数据库是否是支持某特性,比如当前记录的索引数(即结果条数):
int numRows;if(db2.driver()->hasFeature(QSqlDriver::QuerySize)){//是否该特性qDebug()<< "has feature:query size";numRows = query2.size();}else{qDebug() << "no feature:query size";query2.last();numRows = query2.at() + 1;//使用at,需要之前使用quey2.next()遍历所有select搜索后的结果,而使用query2.size()则不需要}//此处执行上面的查询操作,下面的操作才有意义qDebug() << "row number: " << numRows;//指向索引为1的记录,即第二条记录query2.seek(1);//返回当前索引值qDebug() << "current index:" << query2.at();//获得当前行的记录QSqlRecord record = query2.record();//获得记录中"id"和"name"两个字段的值int id = record.value("id").toInt();QString name = record.value("name").toString();qDebug() <<"id" << id << "name:" <<name;//获得索引为1的字段,即第二个字段QSqlField field = record.field(1);//输出字段名和字段值,结果为"name"和"MaLiang"qDebug() << "second field:" << field.name()<< "field value:" << field.value().toString();
- 事务(使数据操作变为原子性)
如果中间有一步sql操作执行出错,则全部sql操作都不执行。
QSqlDatabase db2 = QSqlDatabase::database("connection2");QSqlDatabase::database().transaction();//开始(类似于mutex线程锁)QSqlQuery query(db2);//此语句必须在上面一条语句的后面//执行sql操作QSqlDatabase::database().commit();//结束
2.4.2:使用QSqlQueryModel查询模型
优势:
- 这是基于sql查询的只读模型,编写sql语句变得容易。
文件头:
#include <QSqlQueryModel>
核心代码:
QSqlDatabase db = QSqlDatabase::database("connection1");QSqlQueryModel *model = new QSqlQueryModel(this);model->setQuery("select * from student", db);model->setHeaderData(0, Qt::Horizontal, tr("学号"));model->setHeaderData(1, Qt::Horizontal, tr("姓名"));model->setHeaderData(2, Qt::Horizontal, tr("课程"));QTableView *view = new QTableView(this);view->setModel(model);setCentralWidget(view);
2.4.3:使用QSqlTableModel表格模型⭐
先上结果

优势:
-
编译的代码很容易适应其他的数据源,例如后面如果要使用xml文件来存储数据,只需要更换数据模型。
-
提供了一次只能操作一个sql表的读/写模型,可以浏览和修改独立的sql表,并且只需编写很少的代码,无需了解sql语句。
-
1 准备
头文件:
#include <QSqlTableModel>
QSqlTableModel* model;//创建对象指针
- 2 进行操作:
QSqlDatabase db = QSqlDatabase::database("connection1");model = new QSqlTableModel(this, db);//由于在窗口的类中创建对象,因此实例化对象时,使用this指针(指向操作函数的指针)作为父对象model->setTable("student");model->select();//执行//设置编辑策略model->setEditStrategy(QSqlTableModel::OnManualSubmit);//对所有模型改变立即用到数据库ui->tableView->setModel(model);
-
UI设计:

-
对应槽函数:
// 提交修改按钮
void MainWindow::on_pushButton_clicked()
{// 开始事务操作model->database().transaction();if (model->submitAll()) {if(model->database().commit()) // 提交QMessageBox::information(this, tr("tableModel"),tr("数据修改成功!"));} else {model->database().rollback(); // 回滚QMessageBox::warning(this, tr("tableModel"),tr("数据库错误: %1").arg(model->lastError().text()),QMessageBox::Ok);}
}// 撤销修改按钮
void MainWindow::on_pushButton_2_clicked()
{model->revertAll();
}// 查询按钮,进行筛选
void MainWindow::on_pushButton_5_clicked()
{QString name = ui->lineEdit->text();// 根据姓名进行筛选,一定要使用单引号model->setFilter(QString("name = '%1'").arg(name));model->select();
}// 显示全表按钮
void MainWindow::on_pushButton_6_clicked()
{model->setTable("student");model->select();
}// 按id升序排列按钮
void MainWindow::on_pushButton_7_clicked()
{//id字段,即第0列,升序排列model->setSort(0, Qt::AscendingOrder);model->select();}// 按id降序排列按钮
void MainWindow::on_pushButton_8_clicked()
{model->setSort(0, Qt::DescendingOrder);model->select();}
// 删除选中行按钮
void MainWindow::on_pushButton_4_clicked()
{// 获取选中的行int curRow = ui->tableView->currentIndex().row();// 删除该行model->removeRow(curRow);int ok = QMessageBox::warning(this,tr("删除当前行!"),tr("你确定删除当前行吗?"), QMessageBox::Yes, QMessageBox::No);if(ok == QMessageBox::No){ // 如果不删除,则撤销model->revertAll();} else { // 否则提交,在数据库中删除该行model->submitAll();}}
// 添加记录按钮
void MainWindow::on_pushButton_3_clicked()
{// 获得表的行数int rowNum = model->rowCount();int id = 10;// 添加一行model->insertRow(rowNum);model->setData(model->index(rowNum, 0), id);// 可以直接提交//model->submitAll();
}
三、项目实战
connection.h是负责连接数据库和创建表的头文件:
#ifndef CONNECTION_H
#define CONNECTION_H#include <QMessageBox>
#include <QSqlDatabase>
#include <QSqlQuery>
#include <QDebug>
#pragma execution_character_set("utf-8"); static bool createConnection() {//连接第一个数据库//QMYSQLQSqlDatabase db = QSqlDatabase::addDatabase("QODBC", "connection1");//需要使用的数据库驱动和联检建立的名称(方便建立多个数据库连接【使用不同的数据库时】区分)db.setHostName("127.0.0.1");//连接地址db.setUserName("root");//数据库账户db.setPassword("55667788");//密码db.setPort(3306);//端口//test.dbdb.setDatabaseName("test");//需要用到的数据库if (!db.open()) {//如果数据库连接失败,则弹出//critical(QWidget *parent, const QString &title,//const QString &text,//QMessageBox::StandardButtons buttons = Ok,//QMessageBox::StandardButton defaultButton = NoButton)QMessageBox::critical(0, "Cannot open database","Unable to establish a database connection", QMessageBox::Cancel);return false;}else {QMessageBox::information(NULL, "infor", "link success");}//下面来创建表//如果MySQL数据库中已经存在同名的表,则下面代码不会执行QSqlQuery query2(db);// qDebug() << "connection2:";
//创建表,并插入值query2.exec("create table student (id int primary key,""name varchar(20))");query2.exec("insert into student values(0, 'Mike')");query2.exec("insert into student values(1, 'Lili')");query2.exec("insert into student values(2, 'Jame')");}#endif // CONNECTION_H
Admin是操作数据库的管理界面:

Admin.h:
#pragma once#include <QMainWindow>
#include "ui_Admin.h"
#include <QSqlTableModel>
#pragma execution_character_set("utf-8"); class Admin : public QMainWindow
{Q_OBJECTpublic:Admin(QWidget *parent = nullptr);~Admin();QSqlTableModel* model;//创建对象指针private:Ui::AdminClass ui;private slots:void on_add_clicked();void on_modify_clicked();void on_del_clicked();void on_rollback_clicked();void on_show_all_clicked();
};
Admin.cpp:
#include "Admin.h"
#include <qmessagebox.h>
#include <QSqlDatabase>
#include <QMessageBox>
#include <qsqlerror.h>
#include "connection.h"
#include <QSqlTableModel>Admin::Admin(QWidget *parent): QMainWindow(parent)
{ui.setupUi(this);if (!createConnection()) {return;}QSqlDatabase db = QSqlDatabase::database("connection1");model = new QSqlTableModel(this, db);//由于在窗口的类中创建对象,因此实例化对象时,使用this指针(指向操作函数的指针)作为父对象model->setTable("student");model->select();//执行QTableView* view = new QTableView;view->setModel(model);// 你可以根据你数据库表的实际列名进行设置QHeaderView* header = view->horizontalHeader();header->setModel(model);header->setSectionResizeMode(QHeaderView::Stretch);// 设置显示名称,从0开始计数model->setHeaderData(0, Qt::Horizontal, tr("列名1"));model->setHeaderData(1, Qt::Horizontal, tr("列名2"));model->setHeaderData(2, Qt::Horizontal, tr("列名3"));//设置编辑策略model->setEditStrategy(QSqlTableModel::OnManualSubmit);//对所有模型改变立即用到数据库ui.tableView->setModel(model);}Admin::~Admin()
{}// 添加记录按钮
void Admin::on_add_clicked()
{// 获得表的行数int rowNum = model->rowCount();int id = 10;// 添加一行model->insertRow(rowNum);model->setData(model->index(rowNum, 0), id);// 可以直接提交//model->submitAll();
}// 删除选中行按钮
void Admin::on_del_clicked()
{// 获取选中的行int curRow = ui.tableView->currentIndex().row();// 删除该行model->removeRow(curRow);int ok = QMessageBox::warning(this, tr("删除当前行!"),tr("你确定删除当前行吗?"), QMessageBox::Yes, QMessageBox::No);if (ok == QMessageBox::No){ // 如果不删除,则撤销model->revertAll();}else { // 否则提交,在数据库中删除该行model->submitAll();}}
// 撤销修改按钮
void Admin::on_rollback_clicked()
{model->revertAll();
}// 提交修改按钮
void Admin::on_modify_clicked()
{// 开始事务操作model->database().transaction();if (model->submitAll()) {if (model->database().commit()) // 提交QMessageBox::information(this, tr("tableModel"),tr("数据修改成功!"));}else {model->database().rollback(); // 回滚QMessageBox::warning(this, tr("tableModel"),tr("数据库错误: %1").arg(model->lastError().text()),QMessageBox::Ok);}
}
// 显示全表按钮
void Admin::on_show_all_clicked() {model->setTable("student");model->select();
}
功能正常:

相关文章:
QT C++实践|超详细数据库的连接和增删改查操作|附源码
0:前言 🪧 什么情况需要数据库? 1 大规模的数据需要处理(比如上千上万的数据量)2 需要把数据信息存储起来,无论是本地还是服务上,而不是断电后数据信息就消失了。 如果不是上面的原因化,一般…...
matlab:涉及复杂函数图像的交点求解
matlab:涉及复杂函数图像的交点求解 在MATLAB中求解两个图像的交点是一个常见的需求。本文将通过一个示例,展示如何求解两个图像的交点,并提供相应的MATLAB代码。 画出图像 首先,我们需要绘制两个图像,以便直观地看…...
Unity(第二十二部)官方的反向动力学一般使用商城的IK插件,这个用的不多
反向动力学(Inverse Kinematic,简称IK)是一种通过子节点带动父节点运动的方法。 正向动力学 在骨骼动画中,大多数动画是通过将骨架中的关节角度旋转到预定值来生成的,子关节的位置根据父关节的旋转而改变,这…...
nginx反向代理,获取客户端ip
一、获取客户端ip代码 /*** description: 获取客户端IP* return string*/ public static function getClientIp(){$ip ;if(getenv(HTTP_CLIENT_IP) && strcasecmp(getenv(HTTP_CLIENT_IP),unknown)){$ip getenv(HTTP_CLIENT_IP);}else if(getenv(HTTP_X_FORWARDED_F…...
13 Codeforces Round 886 (Div. 4)G. The Morning Star(简单容斥)
G. The Morning Star 思路:用map记录x,y,以及y-x、yx从前往后统计一遍答案即可公式 a n s c n t [ x ] c n t [ y ] − 2 ∗ c n t [ x , y ] c n t [ y x ] c n t [ y − x ] anscnt[x]cnt[y]-2 * cnt[x,y]cnt[yx]cnt[y-x] anscnt[x]…...
CLion 2023:专注于C和C++编程的智能IDE mac/win版
JetBrains CLion 2023是一款专为C和C开发者设计的集成开发环境(IDE),它集成了许多先进的功能,旨在提高开发效率和生产力。 CLion 2023软件获取 CLion 2023的智能代码编辑器提供了丰富的代码补全和提示功能,使您能够更…...
数据可视化基础与应用-02-基于powerbi实现连锁糕点店数据集的仪表盘制作
总结 本系列是数据可视化基础与应用的第02篇,主要介绍基于powerbi实现一个连锁糕点店数据集的仪表盘制作。 数据集描述 有一个数据集,包含四张工作簿,每个工作簿是一张表,其中可以销售表可以划分为事实表,产品表&am…...
前后端分离Vue+nodejs酒店公寓客房预订管理系统udr7l-java-php-django-springboot
本系统的设计与实现共包含13个表:分别是关于我们信息表,配置文件信息表,公寓信息评论表信息表,公寓入住信息表,公寓退房信息表,公寓信息信息表,公寓预订信息表,系统公告信息表,收藏表…...
VUE打包的dist文件放到后端一起发布
背景 前后端分离开发的项目,在部署时为了方便部署,使用集成部署的方式(即前后端在一起部署的方式) 问题 直接将前端打包好的dist文件夹下的内容,放到后端项目的resource/static目录下,但是在启动访问时发…...
React入门之React_渲染基础用法和class实例写法
渲染元素 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>04元素渲染</title><script src&…...
Git自动忽略dll文件的问题
检查了半天发现是sourcetreee的全局忽略文件导致, 从里面删除dll即可。 我是干脆直接删了全局忽略,太恶心了,如下: #ignore thumbnails created by windows Thumbs.db #Ignore files build by Visual Studio *.exe .vsconfig .s…...
sql中如何实现递归
在SQL中,递归通常是通过使用公用表表达式(Common Table Expressions,CTE)来实现的。CTE允许你定义一个临时的结果集,该结果集可以在一个SELECT、INSERT、UPDATE或DELETE语句的主体中被引用。 递归CTE有两个关键部分&a…...
GPT 的基础 - T(Transformer)
我们知道GPT的含义是: Generative - 生成下一个词 Pre-trained - 文本预训练 Transformer - 基于Transformer架构 我们看到Transformer模型是GPT的基础,这篇博客梳理了一下Transformer的知识点。 BERT: 用于语言理解。(Transformer的Encoder…...
微信小程序 --- 常用样式和组件
常用样式和组件 1. 组件和样式介绍 在开 Web 网站的时候: 页面的结构由 HTML 进行编写,例如:经常会用到 div、p、 span、img、a 等标签 页面的样式由 CSS 进行编写,例如:经常会采用 .class 、#id 、element 等选择…...
深圳智能制造半导体芯片行业源代码防泄密完整解决方案
一、芯片半导体行业防泄密,不能用监控及管控方式来实现,采用管控方式,首先不能主动防御,只能进行事后查询,并且管控方式,不利于嵌入式开发,对于嵌入式开发,不管是采用沙箱隔离或u口禁…...
Unity UI适配规则和对热门游戏适配策略的拆解
前言 本文会介绍一些关于UI适配的基础概念,并且统计了市面上常见的设备的分辨率的情况。同时通过拆解目前市面上较为成功的两款休闲游戏Royal Match和Monopoly GO(两款均为近期游戏付费榜前几的游戏),大致推断出他们的适配策略,以供学习和参…...
嵌入式学习day25 Linux
进程基本概念: 1.进程: 程序:存放在外存中的一段数据组成的文件 进程:是一个程序动态执行的过程,包括进程的创建、进程的调度、进程的消亡 2.进程相关命令: 1.top 动态查看当前系统中的所有进程信息(根据CPU占用率排序&a…...
Oracle数据泵跨大版本迁移数据库
Oracle数据泵跨大版本迁移数据库 source库导出元数据传输dump和数据文件到target库target库导入数据库迁移后的工作 🦈场景:将测试环境中一台Oracle 11g数据库迁移到另一台新搭建的19c数据库中。 🤔分析:由于是跨数据库大版本&…...
如何在Win系统从零开始搭建Z-blog网站,并将本地博客发布到公网可访问
文章目录 1. 前言2. Z-blog网站搭建2.1 XAMPP环境设置2.2 Z-blog安装2.3 Z-blog网页测试2.4 Cpolar安装和注册 3. 本地网页发布3.1. Cpolar云端设置3.2 Cpolar本地设置 4. 公网访问测试5. 结语 1. 前言 想要成为一个合格的技术宅或程序员,自己搭建网站制作网页是绕…...
sawForceDimensionSDK安装,sigma7+ros
force dimension的sdk中没有关于ros,借助开源的sawForceDimensionSDK实现对于数据的封装和可视化,方便后续使用 链接: GitHub - jhu-saw/sawForceDimensionSDK 具体步骤: 安装qt和ros,官网下载Force Dimension SDK …...
Java 语言特性(面试系列1)
一、面向对象编程 1. 封装(Encapsulation) 定义:将数据(属性)和操作数据的方法绑定在一起,通过访问控制符(private、protected、public)隐藏内部实现细节。示例: public …...
定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...
最新SpringBoot+SpringCloud+Nacos微服务框架分享
文章目录 前言一、服务规划二、架构核心1.cloud的pom2.gateway的异常handler3.gateway的filter4、admin的pom5、admin的登录核心 三、code-helper分享总结 前言 最近有个活蛮赶的,根据Excel列的需求预估的工时直接打骨折,不要问我为什么,主要…...
srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...
关于 WASM:1. WASM 基础原理
一、WASM 简介 1.1 WebAssembly 是什么? WebAssembly(WASM) 是一种能在现代浏览器中高效运行的二进制指令格式,它不是传统的编程语言,而是一种 低级字节码格式,可由高级语言(如 C、C、Rust&am…...
【C++从零实现Json-Rpc框架】第六弹 —— 服务端模块划分
一、项目背景回顾 前五弹完成了Json-Rpc协议解析、请求处理、客户端调用等基础模块搭建。 本弹重点聚焦于服务端的模块划分与架构设计,提升代码结构的可维护性与扩展性。 二、服务端模块设计目标 高内聚低耦合:各模块职责清晰,便于独立开发…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...
SQL慢可能是触发了ring buffer
简介 最近在进行 postgresql 性能排查的时候,发现 PG 在某一个时间并行执行的 SQL 变得特别慢。最后通过监控监观察到并行发起得时间 buffers_alloc 就急速上升,且低水位伴随在整个慢 SQL,一直是 buferIO 的等待事件,此时也没有其他会话的争抢。SQL 虽然不是高效 SQL ,但…...
为什么要创建 Vue 实例
核心原因:Vue 需要一个「控制中心」来驱动整个应用 你可以把 Vue 实例想象成你应用的**「大脑」或「引擎」。它负责协调模板、数据、逻辑和行为,将它们变成一个活的、可交互的应用**。没有这个实例,你的代码只是一堆静态的 HTML、JavaScript 变量和函数,无法「活」起来。 …...
OD 算法题 B卷【正整数到Excel编号之间的转换】
文章目录 正整数到Excel编号之间的转换 正整数到Excel编号之间的转换 excel的列编号是这样的:a b c … z aa ab ac… az ba bb bc…yz za zb zc …zz aaa aab aac…; 分别代表以下的编号1 2 3 … 26 27 28 29… 52 53 54 55… 676 677 678 679 … 702 703 704 705;…...
