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

利用可变参数模板,可打印任意参数和参数值。(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)中的参数包。它允许宏接受任意数量的参数。

详细说明

  1. 可变参数宏

    • 在宏定义中,使用 ... 表示可变参数。

    • 使用 __VA_ARGS__ 来引用这些可变参数。

  2. #__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 是一个函数,它至少接受两个参数:

  1. 第一个参数是字符串 "a, b, c",表示变量名。

  2. 后续参数是变量值 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++很好的调式函数)

很酷的应用&#xff1a; &#xff08;1&#xff09; 如何获取可变参数名 代码例子&#xff1a; #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采用多线程架构&#xff0c;线程分为两类&#xff1a; • 工作线程&#xff08;Worker Threads&#xff09;&#xff1a;每个客户端连接到数据库实例时&#xff0c;会创建一个工作线程。工作线程负责处理客户端的SQL请求&#xff0c;执…...

测试工程师Deepseek实战之如何反向PUA它

问: 你是一名资深测试开发工程师 帮我设计一个提效工具&#xff0c;具有以下功能&#xff1a; 1.页面使用PYQT5设计&#xff0c;用两个输入控件&#xff0c;最好是日期类型的控件&#xff0c;第一个日期控件作为开始日期&#xff0c;第二个日期控件作为结束日期&#xff1b;前后…...

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 并配置好环境变量 注&#xff1a;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 适用于轻量级定时任务&#xff0c;基于线程池实现。API 简单&#xff0c;适用于小规模任务调度。 Quartz 强大的 Java 任务调度框架&#xff0c;支持 Cron 表达式、分布式集群、持久化等。适用于复杂调度场景&#xff0…...

ESP32S3读取数字麦克风INMP441的音频数据

ESP32S3 与 INMP441 麦克风模块的集成通常涉及使用 I2S 接口进行数字音频数据的传输。INMP441 是一款高性能的数字麦克风&#xff0c;它通过 I2S 接口输出音频数据。在 Arduino 环境中&#xff0c;ESP32S3 的开发通常使用 ESP-IDF&#xff08;Espressif IoT Development Framew…...

利用后缀表达式构造表达式二叉树的方法

后缀表达式&#xff08;逆波兰表达式&#xff09;是一种将运算符放在操作数之后的表达式表示法。利用后缀表达式构造表达式二叉树的方法主要依赖于栈结构。 转换步骤 初始化 创建一个空栈。 遍历后缀表达式 对后缀表达式的每个符号依次处理&#xff1a; 遇到操作数 如果当前符…...

使用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 是“你只看一次”&#xff08;You Only Look Once, YOLO&#xff09;系列的最新版本&#xff0c;于 2025 年 2 月发布。它引入了注意力机制&#xff0c;提升了检测精度&#xff0c;同时保持了高效的实时性能。在保持速度的同时&#xff0c;显著提升了检测精度。例如&am…...

SQLAlchemy系列教程:如何防止SQL注入

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

1. 树莓派上配置机器人环境(具身智能机器人套件)

1. 安装树莓派系统 镜像下载地址&#xff08;windows/Mac/Ubuntu)&#xff0c;安装Pi5. 2. 环境配置&#xff08;登录Pi系统&#xff09; 2.1 启用 SSH From the Preferences menu, launch Raspberry Pi Configuration. Navigate to the Interfaces tab. Select Enable…...

基于SpringBoot的智慧停车场小程序(源码+论文+部署教程)

运行环境 • 前端&#xff1a;小程序 Vue • 后端&#xff1a;Java • IDE工具&#xff1a;IDEA&#xff08;可自行选择&#xff09; HBuilderX 微信开发者工具 • 技术栈&#xff1a;小程序 SpringBoot Vue MySQL 主要功能 智慧停车场微信小程序主要包含小程序端和…...

【从零开始学习计算机科学】数字逻辑(九)有限状态机

【从零开始学习计算机科学】数字逻辑(九)有限状态机 有限状态机状态机的表示方法有限状态机的Verilog描述有限状态机 有限状态机(简称状态机)相当于一个控制器,它将一项功能的完成分解为若干步,每一步对应于二进制的一个状态,通过预先设计的顺序在各状态之间进行转换,状…...

HarmonyOS Next~鸿蒙系统ArkCompiler跨平台编译技术的革新实践

HarmonyOS Next~鸿蒙系统ArkCompiler跨平台编译技术的革新实践 引言 在万物互联时代&#xff0c;操作系统对编译技术的需求已从单纯的代码转换演变为跨设备协同、高效资源调度与极致性能优化的综合挑战。华为鸿蒙系统&#xff08;HarmonyOS&#xff09;自主研发的ArkCompiler…...

AI大模型概念知多少

什么是大模型&#xff1f;什么是模型参数 1&#xff09;现在的大模型要解决的问题&#xff0c;就是一个序列数据转换的问题&#xff1a; 输入序列 X X[x1 ,x2 ,...,xm ]&#xff0c; 输出序列Y[y1 ,y2 ,…,yn ]&#xff0c;X和Y之间的关系是&#xff1a;YWX。 “大模型”这个词…...

powermock,mock使用笔记

介于日本的形式主义junit4单体测试&#xff0c;特记笔记&#xff0c;以下纯用手机打出来&#xff0c;因为电脑禁止复制粘贴。 pom文件 powermock-module-junit1.7.4 powermock-api-mokcito 1.7.4 spring-test 8 1&#xff0c;测试类头部打注解 RunWith(PowerMockRunner.class…...

基于置换对称性的模型融合:实现凸盆地单盆地理论

【摘要】 一种合并神经网络模型的新方法,通过置换对称性来合并模型。即使在大规模的非凸优化问题中,神经网络损失景观似乎通常只有一个(几乎)封闭的盆地,这在很大程度上归因于隐藏层单元置换对称性。作者介绍了三种算法,用于将一个模型的单元置换为与参考模型对齐,从而…...

把握好自己的节奏, 别让世界成为你的发条匠

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

如何构建你的第一个Python高频交易模型:完整实战指南

如何构建你的第一个Python高频交易模型&#xff1a;完整实战指南 【免费下载链接】High-Frequency-Trading-Model-with-IB A high-frequency trading model using Interactive Brokers API with pairs and mean-reversion in Python 项目地址: https://gitcode.com/gh_mirror…...

如何突破Windows权限壁垒?系统管理专家的秘密武器

如何突破Windows权限壁垒&#xff1f;系统管理专家的秘密武器 【免费下载链接】NSudo [Deprecated, work in progress alternative: https://github.com/M2Team/NanaRun] Series of System Administration Tools 项目地址: https://gitcode.com/gh_mirrors/ns/NSudo 在W…...

【模型手术室】第七篇:模型量化 —— 从 FP16 到 4-bit 的极限压缩与性能翻倍

专栏进度&#xff1a;07 / 10 (微调实战专题) 大模型默认使用 FP16&#xff08;16 位浮点数&#xff09; 存储权重&#xff0c;这意味着每个参数占 2 字节。一个 7B 模型光权重就占 14GB 显存。量化的本质是把这些高精度的数字映射到更小的整数空间&#xff08;如 INT4&#xf…...

Kettle错误处理实战:如何用表输出步骤捕获并存储ETL过程中的异常数据

Kettle错误处理实战&#xff1a;如何用表输出步骤捕获并存储ETL过程中的异常数据 在数据仓库和ETL&#xff08;Extract, Transform, Load&#xff09;流程中&#xff0c;错误处理是确保数据质量的关键环节。Kettle&#xff08;现称Pentaho Data Integration&#xff09;作为一款…...

STC89C52单片机+槽型光耦,手把手教你DIY一个低成本电机转速测量仪

STC89C52单片机槽型光耦DIY电机转速测量仪实战指南 从零搭建低成本测速系统的完整方案 电机转速测量在工业控制、机器人开发、智能小车等领域都是基础但关键的环节。市面上专业测速仪动辄上千元的价格让许多电子爱好者望而却步。其实&#xff0c;利用手头常见的STC89C52单片机…...

OneAgent智能体全球发布会圆满落幕:引领金融AI交易新时代

2026年3月25日&#xff0c;聚焦金融AI领域的盛会《OneAgent智能体全球产品发布会》在中国杭州成功落幕。本次发布会吸引了全球金融科技领域的行业专家、投资机构以及技术爱好者的关注&#xff0c;标志着OneAgent在全球AI金融市场的战略布局正式启动。AI原生对冲交易新物种&…...

从晶体管到CPU:CMOS反相器延迟如何决定你的电脑主频

从晶体管到CPU&#xff1a;CMOS反相器延迟如何决定你的电脑主频 当你按下电脑电源键的瞬间&#xff0c;数十亿个晶体管在芯片上开始协同工作。这些微观开关的切换速度&#xff0c;直接决定了处理器主频的上限。而构成所有数字电路基础的CMOS反相器&#xff0c;其动态响应特性就…...

Node RED实战:5分钟搞定MQTT消息发布与订阅(附EMQX配置)

Node RED与MQTT实战&#xff1a;从零构建物联网消息系统 1. 为什么选择Node RED与MQTT组合&#xff1f; 物联网开发领域一直存在一个核心挑战&#xff1a;如何快速搭建可靠的消息通信系统而不陷入底层协议实现的泥潭。这正是Node RED与MQTT这对黄金组合的价值所在——它们让开发…...

嵌入式AI边缘计算原型:STM32与云端PyTorch模型协同工作流设计

嵌入式AI边缘计算原型&#xff1a;STM32与云端PyTorch模型协同工作流设计 1. 场景需求与痛点分析 在智能家居、工业监测等物联网场景中&#xff0c;我们常常遇到这样的矛盾&#xff1a;边缘设备需要实时响应&#xff0c;但计算能力有限&#xff1b;云端算力强大&#xff0c;但…...

为什么你的STM32F103工程编译失败?可能是启动文件没选对!

为什么你的STM32F103工程编译失败&#xff1f;可能是启动文件没选对&#xff01; 在嵌入式开发领域&#xff0c;STM32系列微控制器因其出色的性能和丰富的外设资源而广受欢迎。然而&#xff0c;即使是经验丰富的开发者&#xff0c;在STM32F103项目开发过程中也难免会遇到各种编…...