利用可变参数模板,可打印任意参数和参数值。(C++很好的调式函数)
很酷的应用:
(1) 如何获取可变参数名
代码例子:
#define _test(...) (test_t(#__VA_ARGS__, __VA_ARGS__))template<typename... Args>
void test_t(const char* names, Args... args)
{std::cout << names << "\n";
}
__VA_ARGS__
是 C/C++ 宏定义中的一个特殊标识符,用于表示 可变参数宏(Variadic Macros)中的参数包。它允许宏接受任意数量的参数。
详细说明
-
可变参数宏:
-
在宏定义中,使用
...
表示可变参数。 -
使用
__VA_ARGS__
来引用这些可变参数。
-
-
#__VA_ARGS__
:-
#
是字符串化运算符,将宏参数转换为字符串。 -
#__VA_ARGS__
将可变参数包中的所有参数转换为一个字符串。
-
#define _test(...) (test_t(#__VA_ARGS__, __VA_ARGS__))
这条语句定义了一个宏 _test
,它接受可变数量的参数(通过 ...
表示),并将这些参数传递给一个函数 test_t
:
1. 宏定义的基本结构
-
_test
是宏的名称。 -
(...)
表示宏可以接受任意数量的参数。 -
test_t(#__VA_ARGS__, __VA_ARGS__)
是宏的展开内容。
2. __VA_ARGS__
的作用
-
__VA_ARGS__
是 C/C++ 中的特殊标识符,用于表示可变参数宏中的参数包。 -
它允许宏接受任意数量的参数。
3. #__VA_ARGS__
的作用
-
#
是字符串化运算符,将宏参数转换为字符串。 -
#__VA_ARGS__
将可变参数包中的所有参数转换为一个字符串。
4. 宏的展开逻辑
假设调用 _test(a, b, c)
,宏会展开为:
(test_t("a, b, c", a, b, c))
-
#__VA_ARGS__
将a, b, c
转换为字符串"a, b, c"
。 -
__VA_ARGS__
展开为a, b, c
。
5. test_t
函数的作用
test_t
是一个函数,它至少接受两个参数:
-
第一个参数是字符串
"a, b, c"
,表示变量名。 -
后续参数是变量值
a, b, c
。
test_t
的具体实现需要根据需求编写。例如,它可以用于打印变量名和变量值,或者进行其他处理。
下面看一下例子:
#define _test(...) (test_t(#__VA_ARGS__, __VA_ARGS__))template<typename... Args>
void test_t(const char* names, Args... args)
{std::cout << names << "\n";
}int main(int argc, char* argv[])
{QApplication a(argc, argv); //注意,这里是QApplication ga.setStdLocaleForUTF8();int i = 5;double c = 3.14;std::string s = "abcd";_test(i, c,s);return a.exec();
}
运行结果:
请注意变量 i 和 c 中间是有一个空格的,这跟你的书写格式有关:
例如:
(2)如何展开参数包。
#define _test(...) (test_t(#__VA_ARGS__, __VA_ARGS__))template<typename... Args>
void test_t(const char* names, Args... args)
{auto arrNames = _string(names).split(','); //拆分变量列表for (auto s : arrNames) {std::wcout << _t("s=") << s << _t("\n");}
}int main(int argc, char* argv[])
{QApplication a(argc, argv); //注意,这里是QApplication ga.setStdLocaleForUTF8(); int i = 5;double c = 3.14;std::string s = "abcd";_test(i,c,s);return a.exec();
}
输出结果:
(3)如何获取变量的值,这里只介绍C++17及其以上的写法。
#define _test(...) (test_t(#__VA_ARGS__, __VA_ARGS__))template<typename... Args>
void test_t(const char* names, Args... args)
{auto arrNames = _string(names).split(','); //拆分变量列表int i = 0;// 使用折叠表达式展开参数包,只支持C++17及其以上//-------------------------------------------------------------------------((std::wcout << arrNames[i],std::cout << "=",std::cout << args << "\n", ++i //下一个参数),...);}int main(int argc, char* argv[])
{QApplication a(argc, argv); //注意,这里是QApplication ga.setStdLocaleForUTF8(); int i = 5;double c = 3.14;std::string s = "abcd";_test(i,c,s);return a.exec();
}
运行结果:
(4)应用例子:
//在控制台打印出n个变量名和变量值
#define _pns(...) (_cout << _generateString(#__VA_ARGS__, __VA_ARGS__))
//在窗口中显示n个变量名和变量值
#define _pnw(...) (qt.showText(_generateString(#__VA_ARGS__, __VA_ARGS__)))
int main(int argc, char* argv[])
{QApplication a(argc, argv); //注意,这里是QApplication ga.setStdLocaleForUTF8(); QWidget w;w.resize(500, 800);int i = 5;double d = 3.14;std::string s = "abcd";_Color c;_Font f;_pns(i, d, s, w.geometry(), c, f);_pnw(i, d, s, w.geometry(), c, f);return a.exec();
}
结果:
下面是代码:
#if _c17_
/*-------------------------------------------- - 源程序(由DeepSeek提供)
// 辅助函数,生成变量名和值的字符串
template<typename... Args>
std::string _generateString(const char* names, Args... args) {std::string result;std::ostringstream oss;std::istringstream iss(names);std::string name;((oss << (iss >> name ? name : "") << " = " << args << "<br>"), ...);return oss.str();
}
*//// <summary>
/// 参数名 = 参数值
/// </summary>
/// <typeparam name="...Args"></typeparam>
/// <param name="names"></param>
/// <param name="...args"></param>
/// <returns></returns>
/// 创建时间:2025-03-07 最后一次修改时间:2025-03-07 (已测试)
template<typename... Args>
_string _generateString(const char* names, Args... args) {_string result;using namespace lf;auto arrNames = _string(names).split(','); //拆分变量列表int i = 0;const char* namePtr = names;//std::cout << "names=" << names << "\n";// 使用折叠表达式展开参数包//((result += std::string(namePtr) + " = " + std::to_string(args) + "<br>", namePtr = nullptr), ...);// 使用折叠表达式展开参数包,只支持C++17及其以上//-------------------------------------------------------------------------((// names = i, d, f, w.geometry(),用逗号和一个空格分隔,是固定的。//result.append(_string(typeid(args).name())),//result.append(_t("<br>")),//result.append(_t("----------")),//result.append(_t("<br>")),//name = _string(namePtr).left(_t(", ")),result.add(arrNames[i].trim()), result.append(_t("=")),result.append(_tostr(args)),//result.append(_t("<br>")), //如果是纯文本 <br> 替代成 \nresult.append(_t("\n")),++i //下一个参数),...);//-------------------------------------------------------------------------return result;
}
#else
/*---------------------------------------------源程序(由DeepSeek提供)
// 递归终止条件
std::string _generateStringHelper(const char* namePtr) {return "";
}// 递归展开参数包
template<typename T, typename... Args>
std::string _generateStringHelper(const char* namePtr, T arg, Args... args) {std::string result;// 提取变量名std::string name;while (*namePtr && *namePtr != ',') {name += *namePtr++;}if (*namePtr == ',') namePtr++; // 跳过逗号while (*namePtr == ' ') namePtr++; // 跳过空格// 拼接变量名和值result += name + " = " + std::to_string(arg) + "<br>";// 递归处理剩余参数result += _generateStringHelper(namePtr, args...);return result;
}template<typename... Args>
std::string _generateString(const char* names, Args... args) {return _generateStringHelper(names, args...);
}
*/// 递归终止条件
_string _generateStringHelper(const char* namePtr) {return _t("");
}// 递归展开参数包
template<typename T, typename... Args>
_string _generateStringHelper(const char* namePtr, T arg, Args... args) {_string result;_pn(namePtr);// 提取变量名_string name;while (*namePtr && *namePtr != ',') {name.append(*namePtr++);}if (*namePtr == ',') namePtr++; // 跳过逗号while (*namePtr == ' ') namePtr++; // 跳过空格// 拼接变量名和值result += name + _t(" = ") + _tostr(arg) + _t("<br>");// 递归处理剩余参数result += _generateStringHelper(namePtr, args...);return result;
}template<typename... Args>
_string _generateString(const char* names, Args... args) {return _generateStringHelper(names, args...);
}#endif
这里有个关键函数,_tostr上次已介绍过:
_tostr
/// <summary>
///
/// </summary>
/// <param name="pt"></param>
/// <param name="sTypeName"></param>
/// <returns></returns>
/// 创建时间:2025-02-16 最后一次修改时间:2025-02-16
_string _tostr(void* pt, const char* sTypeName);/// <summary>
///
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="t"></param>
/// <returns></returns>
/// 创建时间:2025-02-16 最后一次修改时间:2025-02-16
template<class T>
inline _string _tostr(const T& t) {return _tostr((void*)(&t), typeid(t).name());
}
_string _tostr(void* pt, const char* sTypeName)
{_StrA sType = sTypeName;_string sResult(_t(""), 100);if (sType == typeid(char*).name() || sType == typeid(const char*).name()) {char** ppC = ((char**)(pt)); //指向指针的指针sResult.add(_string::qt_fromUtf8(*ppC));}else if (sType == typeid(wchar_t*).name() || sType == typeid(const wchar_t*).name()) {wchar_t** ppC = ((wchar_t**)(pt)); //指向指针的指针sResult.add(*ppC);}
#ifdef _QT_ //---------------------------------------------------------_QT_ else if (sType == typeid(QStringList).name()) {QStringList* data = (QStringList*)pt;if (data->size() == 0)sResult.add(_t("{}"));else {sResult.add(_t("{"));for (int i = 0; i < data->size() - 1; ++i) {sResult.add(_t("\""));sResult.add(data->at(i).toStdWString());sResult.add(_t("\","));}sResult.add(_t("\""));sResult.add(data->last().toStdWString());sResult.add(_t("\""));sResult.add(_t("}"));}}else if (sType == typeid(QString).name()) {sResult.add(_t("\""));sResult.add((*((QString*)(pt))).toStdWString());sResult.add(_t("\""));}else if (sType == typeid(QDateTime).name()) {return QtDateTimeToString((QDateTime*)pt);}else if (sType == typeid(QFont).name()) {QFont* pf = (QFont*)(pt);sResult.add(pf->toString().toStdWString());}else if (sType == typeid(QByteArray).name()) {QByteArray* pb = (QByteArray*)(pt);sResult.add(_string::fromOnlyData(pb->data(), pb->size()));}else if (sType == typeid(QColor).name()) {QColor* pc = (QColor*)(pt);//sResult.add(_string::fromOnlyData(pb->data(), pb->size()));/*alpha:透明度。red:红色分量。green:绿色分量。blue:蓝色分量。pad:填充字段。*/_Color c;c._a = pc->alpha();c._r = pc->red();c._g = pc->green();c._b = pc->blue();sResult.add(c.toString());}else if (sType == typeid(QSize).name()) {QSize* ps = (QSize*)(pt);sResult.add(_t("("));sResult.add(_Math::intToStrForBaseN(ps->width()));sResult.add(_t(","));sResult.add(_Math::intToStrForBaseN(ps->height()));sResult.add(_t(")"));}else if (sType == typeid(QRect).name()){QRect* pr = (QRect*)(pt);sResult.add(_t("("));sResult.add(_Math::intToStrA(pr->left()));sResult.add(_t(","));sResult.add(_Math::intToStrA(pr->top()));sResult.add(_t(","));sResult.add(_Math::intToStrA(pr->bottom()));sResult.add(_t(","));sResult.add(_Math::intToStrA(pr->right()));sResult.add(_t(")"));}
#endif // ---------------------------------------------------------_lf_ else if (sType == typeid(_StrListW).name()) {_StrListW* data = (_StrListW*)pt;if (data->size() == 0)sResult.add(_t("{}"));else {sResult.add(_t("{"));for (int i = 0; i < data->size() - 1; ++i) {sResult.add(_t("\""));sResult.add(data->at(i).qt_toStdWString());sResult.add(_t("\","));}sResult.add(_t("\""));sResult.add(data->last()->data.qt_toStdWString());sResult.add(_t("\""));sResult.add(_t("}"));}}else if (sType == typeid(_Color).name()) {_Color* pc = ((_Color*)(pt));sResult.add(pc->toString());}else if (sType == typeid(_Font).name()) {_Font* pf = ((_Font*)(pt));sResult.add(pf->toString());}else if (sType == typeid(_GuiTreeNodeEncryptionType).name()) {_GuiTreeNodeEncryptionType* ptnt = ((_GuiTreeNodeEncryptionType*)(pt));if (*ptnt == _GuiTreeNodeEncryptionType::noEncryption) {sResult.add(_t("_GuiTreeNodeEncryptionType::noEncryption"));}else if (*ptnt == _GuiTreeNodeEncryptionType::simpleEncryption) {sResult.add(_t("_GuiTreeNodeEncryptionType::simpleEncryption"));}else if (*ptnt == _GuiTreeNodeEncryptionType::aesEncryption) {sResult.add(_t("_GuiTreeNodeEncryptionType::aesEncryption"));}else if (*ptnt == _GuiTreeNodeEncryptionType::desEncryption) {sResult.add(_t("_GuiTreeNodeEncryptionType::desEncryption"));}}else if (sType == typeid(int).name()) {int *pNum = ((int*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(float).name()) {float* pNum = ((float*)(pt));sResult.add(_Math::doubleToStr(*pNum));}else if (sType == typeid(double).name()) {double *pNum = ((double*)(pt));sResult.add(_Math::doubleToStr(*pNum));}else if (sType == typeid(__int64).name()) {__int64 *pNum = ((__int64*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_byte).name()) {_byte *pNum = ((_byte*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_int8).name()) {_int8 *pNum = ((_int8*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_int16).name()) {_int16 *pNum = ((_int16*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_int32).name()) {_int16 *pNum = ((_int16*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_int64).name()) {_int64 *pNum = ((_int64*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_uint8).name()) {_uint8 *pNum = ((_uint8*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_uint16).name()) {_uint16 *pNum = ((_uint16*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_uint32).name()) {_uint32 *pNum = ((_uint32*)(pt));sResult.add(_Math::intToStr(*pNum));}else if (sType == typeid(_uint64).name()) {_uint64* pNum = ((_uint64*)(pt));sResult.add(std::to_wstring(*pNum));}else if (sType == typeid(std::string).name()) {std::string* ps = ((std::string*)(pt));sResult.add(_string::qt_fromStdString(*ps));}else if (sType == typeid(std::wstring).name()) {std::wstring* ps = ((std::wstring*)(pt));sResult.add(_string::qt_fromStdWString(*ps));}else if (sType == typeid(_string).name()) {_string* ps = ((_string*)(pt));sResult.add(*ps);}else if (sType.indexOf("char const [") != -1) {//char const [9]const char* ps = (char*)(pt);sResult.add(_string::qt_fromUtf8(ps));}else if (sType.indexOf("wchar_t const [") != -1) {//char const [9]const wchar_t* ps = (wchar_t*)(pt);sResult.add(ps);}else if (sType == typeid(bool).name()) {bool* pb = (bool*)(pt);if (*pb)sResult.add(_t("true"));elsesResult.add(_t("false"));}else if (sType.indexOf("lf::_DList") != -1) //class lf::_DList<class lf::_StrW>{_Object* po = (_Object*)(pt);return po->toString();}else if (sType == typeid(_Object).name()) {_Object* po = (_Object*)(pt);return po->toString();}else {sResult.add(sType);}return sResult;
}
相关文章:

利用可变参数模板,可打印任意参数和参数值。(C++很好的调式函数)
很酷的应用: (1) 如何获取可变参数名 代码例子: #define _test(...) (test_t(#__VA_ARGS__, __VA_ARGS__))template<typename... Args> void test_t(const char* names, Args... args) {std::cout << names <<…...
Yashan DB 体系结构
一、体系结构概况 1.1 线程管理 YashanDB采用多线程架构,线程分为两类: • 工作线程(Worker Threads):每个客户端连接到数据库实例时,会创建一个工作线程。工作线程负责处理客户端的SQL请求,执…...

测试工程师Deepseek实战之如何反向PUA它
问: 你是一名资深测试开发工程师 帮我设计一个提效工具,具有以下功能: 1.页面使用PYQT5设计,用两个输入控件,最好是日期类型的控件,第一个日期控件作为开始日期,第二个日期控件作为结束日期;前后…...
Windows系统中在VSCode上配置CUDA环境
前置步骤 安装符合GPU型号的CUDA Toolkit 配置好 nvcc 环境变量 安装 Visual Studio 参考https://blog.csdn.net/Cony_14/article/details/137510909 VSCode 安装插件 Nsight Visual Studio Code Editionvscode-cudacpp 安装 cmake 并配置好环境变量 注:Windows 端…...
React Native 0.76 升级后 APK 体积增大的原因及优化方案
在将 React Native 从 0.71 升级到 0.76 后,打包体积从 40 多 MB 增加到了 80 MB。经过一系列排查和优化,最终找到了解决方案,并将优化过程整理如下。 1. React Native 0.76 体积增大的可能原因 (1) 新架构默认启用 React Native 0.76 默认启用了 New Architecture(新架…...

pycharm找不到conda可执行文件
conda 24.9.2 在pycharm的右下角就可以切换python解释器了...
定时任务框架
常用定时任务框架 JDK 自带的 ScheduledExecutorService 适用于轻量级定时任务,基于线程池实现。API 简单,适用于小规模任务调度。 Quartz 强大的 Java 任务调度框架,支持 Cron 表达式、分布式集群、持久化等。适用于复杂调度场景࿰…...

ESP32S3读取数字麦克风INMP441的音频数据
ESP32S3 与 INMP441 麦克风模块的集成通常涉及使用 I2S 接口进行数字音频数据的传输。INMP441 是一款高性能的数字麦克风,它通过 I2S 接口输出音频数据。在 Arduino 环境中,ESP32S3 的开发通常使用 ESP-IDF(Espressif IoT Development Framew…...
利用后缀表达式构造表达式二叉树的方法
后缀表达式(逆波兰表达式)是一种将运算符放在操作数之后的表达式表示法。利用后缀表达式构造表达式二叉树的方法主要依赖于栈结构。 转换步骤 初始化 创建一个空栈。 遍历后缀表达式 对后缀表达式的每个符号依次处理: 遇到操作数 如果当前符…...

使用express创建服务器保存数据到mysql
创建数据库和表结构 CREATE DATABASE collect;USE collect;CREATE TABLE info (id int(11) NOT NULL AUTO_INCREMENT,create_date bigint(20) DEFAULT NULL COMMENT 时间,type varchar(20) DEFAULT NULL COMMENT 数据分类,text_value text COMMENT 内容,PRIMARY KEY (id) ) EN…...

YOLOv12本地部署教程——42%速度提升,让高效目标检测触手可及
YOLOv12 是“你只看一次”(You Only Look Once, YOLO)系列的最新版本,于 2025 年 2 月发布。它引入了注意力机制,提升了检测精度,同时保持了高效的实时性能。在保持速度的同时,显著提升了检测精度。例如&am…...

SQLAlchemy系列教程:如何防止SQL注入
SQL注入是一种常见的安全漏洞,它允许攻击者通过应用程序的SQL查询操纵数据库。使用ORM工具(如SQLAlchemy)提供的内置功能可以帮助减轻这些风险。本教程将指导您完成保护SQLAlchemy查询的实践。 了解SQL注入 当攻击者能够通过用户输入插入或操…...

1. 树莓派上配置机器人环境(具身智能机器人套件)
1. 安装树莓派系统 镜像下载地址(windows/Mac/Ubuntu),安装Pi5. 2. 环境配置(登录Pi系统) 2.1 启用 SSH From the Preferences menu, launch Raspberry Pi Configuration. Navigate to the Interfaces tab. Select Enable…...

基于SpringBoot的智慧停车场小程序(源码+论文+部署教程)
运行环境 • 前端:小程序 Vue • 后端:Java • IDE工具:IDEA(可自行选择) HBuilderX 微信开发者工具 • 技术栈:小程序 SpringBoot Vue MySQL 主要功能 智慧停车场微信小程序主要包含小程序端和…...

【从零开始学习计算机科学】数字逻辑(九)有限状态机
【从零开始学习计算机科学】数字逻辑(九)有限状态机 有限状态机状态机的表示方法有限状态机的Verilog描述有限状态机 有限状态机(简称状态机)相当于一个控制器,它将一项功能的完成分解为若干步,每一步对应于二进制的一个状态,通过预先设计的顺序在各状态之间进行转换,状…...
HarmonyOS Next~鸿蒙系统ArkCompiler跨平台编译技术的革新实践
HarmonyOS Next~鸿蒙系统ArkCompiler跨平台编译技术的革新实践 引言 在万物互联时代,操作系统对编译技术的需求已从单纯的代码转换演变为跨设备协同、高效资源调度与极致性能优化的综合挑战。华为鸿蒙系统(HarmonyOS)自主研发的ArkCompiler…...
AI大模型概念知多少
什么是大模型?什么是模型参数 1)现在的大模型要解决的问题,就是一个序列数据转换的问题: 输入序列 X X[x1 ,x2 ,...,xm ], 输出序列Y[y1 ,y2 ,…,yn ],X和Y之间的关系是:YWX。 “大模型”这个词…...
powermock,mock使用笔记
介于日本的形式主义junit4单体测试,特记笔记,以下纯用手机打出来,因为电脑禁止复制粘贴。 pom文件 powermock-module-junit1.7.4 powermock-api-mokcito 1.7.4 spring-test 8 1,测试类头部打注解 RunWith(PowerMockRunner.class…...
基于置换对称性的模型融合:实现凸盆地单盆地理论
【摘要】 一种合并神经网络模型的新方法,通过置换对称性来合并模型。即使在大规模的非凸优化问题中,神经网络损失景观似乎通常只有一个(几乎)封闭的盆地,这在很大程度上归因于隐藏层单元置换对称性。作者介绍了三种算法,用于将一个模型的单元置换为与参考模型对齐,从而…...

把握好自己的节奏, 别让世界成为你的发条匠
我见过凌晨两点还在回复工作群消息的职场妈妈,也见过凌晨三点抱着手机刷短视频的年轻人。 地铁站台的上班族永远在狂奔,连刚会走路的小孩都被早教班塞满了日程表。 现如今生活节奏快,像一只巨大的发条,每个人都被拧得紧紧的&#…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...

解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...

visual studio 2022更改主题为深色
visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中,选择 环境 -> 常规 ,将其中的颜色主题改成深色 点击确定,更改完成...

【第二十一章 SDIO接口(SDIO)】
第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...
VTK如何让部分单位不可见
最近遇到一个需求,需要让一个vtkDataSet中的部分单元不可见,查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行,是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示,主要是最后一个参数,透明度…...

如何在最短时间内提升打ctf(web)的水平?
刚刚刷完2遍 bugku 的 web 题,前来答题。 每个人对刷题理解是不同,有的人是看了writeup就等于刷了,有的人是收藏了writeup就等于刷了,有的人是跟着writeup做了一遍就等于刷了,还有的人是独立思考做了一遍就等于刷了。…...

2025季度云服务器排行榜
在全球云服务器市场,各厂商的排名和地位并非一成不变,而是由其独特的优势、战略布局和市场适应性共同决定的。以下是根据2025年市场趋势,对主要云服务器厂商在排行榜中占据重要位置的原因和优势进行深度分析: 一、全球“三巨头”…...

基于SpringBoot在线拍卖系统的设计和实现
摘 要 随着社会的发展,社会的各行各业都在利用信息化时代的优势。计算机的优势和普及使得各种信息系统的开发成为必需。 在线拍卖系统,主要的模块包括管理员;首页、个人中心、用户管理、商品类型管理、拍卖商品管理、历史竞拍管理、竞拍订单…...

Windows安装Miniconda
一、下载 https://www.anaconda.com/download/success 二、安装 三、配置镜像源 Anaconda/Miniconda pip 配置清华镜像源_anaconda配置清华源-CSDN博客 四、常用操作命令 Anaconda/Miniconda 基本操作命令_miniconda创建环境命令-CSDN博客...
SQL Server 触发器调用存储过程实现发送 HTTP 请求
文章目录 需求分析解决第 1 步:前置条件,启用 OLE 自动化方式 1:使用 SQL 实现启用 OLE 自动化方式 2:Sql Server 2005启动OLE自动化方式 3:Sql Server 2008启动OLE自动化第 2 步:创建存储过程第 3 步:创建触发器扩展 - 如何调试?第 1 步:登录 SQL Server 2008第 2 步…...