【学写LibreCAD】1 LibreCAD主程序
一、源码
- 头文件:
#ifndef MAIN_H
#define MAIN_H#include<QStringList>#define STR(x) #x
#define XSTR(x) STR(x)/*** @brief handleArgs* @param argc cli argument counter from main()* @param argv cli arguments from main()* @param argClean a list of indices to be ignored* @return*/
QStringList handleArgs(int argc, char** argv, const QList<int>& argClean);/*** @brief LCReleaseLabel return a label for the current release based on LC_VERSION in src.pro* @return "Release Candidate" - if LC_VERSION contains rc;* "BETA" - if LC_VERSION contains beta* "ALPHA" - if LC_VERSION contains alpha*/
QString LCReleaseLabel();#endif
程序文件:
#include <clocale>#include <QApplication>
#include <QByteArray>
#include <QDebug>
#include <QFileInfo>
#include <QMessageBox>
#include <QPainter>
#include <QPixmap>
#include <QSettings>
#include <QSplashScreen>#include "console_dxf2pdf.h"
#include "console_dxf2png.h"
#include "lc_application.h"
#include "main.h"
#include "qc_applicationwindow.h"
#include "qg_dlginitial.h"
#include "rs_debug.h"
#include "rs_fontlist.h"
#include "rs_patternlist.h"
#include "rs_settings.h"
#include "rs_system.h"namespace{void restoreWindowGeometry(QC_ApplicationWindow& appWin, QSettings& settings);
// update splash for alpha/beta names)void updateSplash(const std::unique_ptr<QSplashScreen>& splash);
}
/*** Main. 创建应用程序窗口。*/// fixme - sand - refactor and split to several specialized functions
int main(int argc, char** argv){QT_REQUIRE_VERSION(argc, argv, "5.2.1"); //确保 LibreCAD 在 Qt 5.2.1 或更高版本上运行//检查前两个参数,以决定我们是要将librecad作为控制台dxf2pdf还是dxf2png工具运行。在Linux上,我们可以创建一个指向librecad可执行文件的链接,并将其命名为dxf2pdf。因此,我们可以运行以下任一操作://librecad dxf2pdf[选项]...//或者只是:dxf2pdf[选项]...for (int i = 0; i < qMin(argc, 2); i++) {QString arg(argv[i]);if (i == 0) {arg = QFileInfo(QFile::decodeName(argv[i])).baseName();}if (arg.compare("dxf2pdf") == 0) {return console_dxf2pdf(argc, argv);}if (arg.compare("dxf2png") == 0 || arg == "dxf2svg") {return console_dxf2png(argc, argv);}}RS_DEBUG->setLevel(RS_Debug::D_WARNING);LC_Application app(argc, argv);QCoreApplication::setOrganizationName("LibreCAD");QCoreApplication::setApplicationName("LibreCAD");QCoreApplication::setApplicationVersion(XSTR(LC_VERSION));RS_Settings::init(app.organizationName(), app.applicationName());QGuiApplication::setDesktopFileName("librecad.desktop");QSettings settings; // fixme - direct invocation of settingsbool first_load = settings.value("Startup/FirstLoad", 1).toBool();const QString lpDebugSwitch0("-d"),lpDebugSwitch1("--debug") ;const QString help0("-h"), help1("--help");bool allowOptions=true;QList<int> argClean;for (int i=0; i<argc; i++){QString argstr(argv[i]);if(allowOptions&&QString::compare("--", argstr)==0){allowOptions=false;continue;}if (allowOptions && (help0.compare(argstr, Qt::CaseInsensitive)==0 ||help1.compare(argstr, Qt::CaseInsensitive)==0 )){qDebug()<<"Usage: librecad [command] <options> <dxf file>";qDebug()<<"";qDebug()<<"Commands:";qDebug()<<"";qDebug()<<" dxf2pdf\tRun librecad as console dxf2pdf tool. Use -h for help.";qDebug()<<" dxf2png\tRun librecad as console dxf2png tool. Use -h for help.";qDebug()<<" dxf2svg\tRun librecad as console dxf2svg tool. Use -h for help.";qDebug()<<"";qDebug()<<"Options:";qDebug()<<"";qDebug()<<" -h, --help\tdisplay this message";qDebug()<<" -d, --debug <level>";qDebug()<<"";RS_DEBUG->print( RS_Debug::D_NOTHING, "possible debug levels:");RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Nothing", RS_Debug::D_NOTHING);RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Critical", RS_Debug::D_CRITICAL);RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Error", RS_Debug::D_ERROR);RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Warning", RS_Debug::D_WARNING);RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Notice", RS_Debug::D_NOTICE);RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Informational", RS_Debug::D_INFORMATIONAL);RS_DEBUG->print( RS_Debug::D_NOTHING, " %d Debugging", RS_Debug::D_DEBUGGING);exit(0);}if (allowOptions&& (argstr.startsWith(lpDebugSwitch0, Qt::CaseInsensitive) ||argstr.startsWith(lpDebugSwitch1, Qt::CaseInsensitive) )){argClean<<i;// to control the level of debugging output use --debug with level 0-6, e.g. --debug3// for a list of debug levels use --debug?// if no level follows, the debugging level is setargstr.remove(QRegularExpression("^"+lpDebugSwitch0));argstr.remove(QRegularExpression("^"+lpDebugSwitch1));char level;if(argstr.size()==0){if(i+1<argc){if(QRegularExpression(R"(\d*)").match(argv[i+1]).hasMatch()){++i;qDebug()<<"reading "<<argv[i]<<" as debugging level";level=argv[i][0];argClean<<i;} else {level = '3';}}else {level = '3'; //default to D_WARNING}}else {level = argstr.toStdString()[0];}switch(level){case '?' : {RS_DEBUG->print(RS_Debug::D_NOTHING, "possible debug levels:");RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Nothing", RS_Debug::D_NOTHING);RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Critical", RS_Debug::D_CRITICAL);RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Error", RS_Debug::D_ERROR);RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Warning", RS_Debug::D_WARNING);RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Notice", RS_Debug::D_NOTICE);RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Informational", RS_Debug::D_INFORMATIONAL);RS_DEBUG->print(RS_Debug::D_NOTHING, " %d Debugging", RS_Debug::D_DEBUGGING);return 0;}case '0' + RS_Debug::D_NOTHING : {RS_DEBUG->setLevel(RS_Debug::D_NOTHING);break;}case '0' + RS_Debug::D_CRITICAL : {RS_DEBUG->setLevel(RS_Debug::D_CRITICAL);break;}case '0' + RS_Debug::D_ERROR : {RS_DEBUG->setLevel(RS_Debug::D_ERROR);break;}case '0' + RS_Debug::D_WARNING : {RS_DEBUG->setLevel(RS_Debug::D_WARNING);break;}case '0' + RS_Debug::D_NOTICE : {RS_DEBUG->setLevel(RS_Debug::D_NOTICE);break;}case '0' + RS_Debug::D_INFORMATIONAL : {RS_DEBUG->setLevel(RS_Debug::D_INFORMATIONAL);break;}case '0' + RS_Debug::D_DEBUGGING : {RS_DEBUG->setLevel(RS_Debug::D_DEBUGGING);break;}default : {RS_DEBUG->setLevel(RS_Debug::D_DEBUGGING);break;}}}}RS_DEBUG->print("param 0: %s", argv[0]);QFileInfo prgInfo( QFile::decodeName(argv[0]) );QString prgDir(prgInfo.absolutePath());RS_SYSTEM->init(app.applicationName(), app.applicationVersion(), XSTR(QC_APPDIR), prgDir);// parse command line arguments that might not need a launched program:QStringList fileList = handleArgs(argc, argv, argClean);QString unit = settings.value("Defaults/Unit", "Invalid").toString();// show initial config dialog:if (first_load){RS_DEBUG->print("main: show initial config dialog..");QG_DlgInitial di(nullptr);QPixmap pxm(":/main/intro_librecad.png");di.setPixmap(pxm);if (di.exec()) {unit = LC_GET_ONE_STR("Defaults", "Unit", "None");}RS_DEBUG->print("main: show initial config dialog: OK");}auto splash = std::make_unique<QSplashScreen>();bool show_splash = settings.value("Startup/ShowSplash", 1).toBool();if (show_splash){updateSplash(splash);app.processEvents();RS_DEBUG->print("main: splashscreen: OK");}RS_DEBUG->print("main: init fontlist..");RS_FONTLIST->init();RS_DEBUG->print("main: init fontlist: OK");RS_DEBUG->print("main: init patternlist..");RS_PATTERNLIST->init();RS_DEBUG->print("main: init patternlist: OK");RS_DEBUG->print("main: loading translation..");settings.beginGroup("Appearance");QString lang = settings.value("Language", "en").toString();QString langCmd = settings.value("LanguageCmd", "en").toString();settings.endGroup();RS_SYSTEM->loadTranslation(lang, langCmd);RS_DEBUG->print("main: loading translation: OK");RS_DEBUG->print("main: creating main window..");QC_ApplicationWindow& appWin = *QC_ApplicationWindow::getAppWindow();
#ifdef Q_OS_MACapp.installEventFilter(&appWin);
#endifRS_DEBUG->print("main: setting caption");appWin.setWindowTitle(app.applicationName());RS_DEBUG->print("main: show main window");settings.beginGroup("Defaults");if( !settings.contains("UseQtFileOpenDialog")) {
#ifdef Q_OS_LINUX// on Linux don't use native file dialog// because of case insensitive filters (issue #791)settings.setValue("UseQtFileOpenDialog", QVariant(1));
#elsesettings.setValue("UseQtFileOpenDialog", QVariant(0));
#endif}settings.endGroup();if (!first_load) {restoreWindowGeometry(appWin, settings);}bool maximize = settings.value("Startup/Maximize", 0).toBool();if (maximize || first_load) {appWin.showMaximized();}else {appWin.show();}RS_DEBUG->print("main: set focus");appWin.setFocus();RS_DEBUG->print("main: creating main window: OK");if (show_splash){RS_DEBUG->print("main: updating splash");splash->raise();splash->showMessage(QObject::tr("Loading..."),Qt::AlignRight|Qt::AlignBottom, Qt::black);RS_DEBUG->print("main: processing events");qApp->processEvents();RS_DEBUG->print("main: updating splash: OK");}// Set LC_NUMERIC so that entering numeric values uses . as the decimal separatorsetlocale(LC_NUMERIC, "C");RS_DEBUG->print("main: loading files..");
#ifdef Q_OS_MAC// get the file list from LC_ApplicationfileList << app.fileList();
#endif// reopen files that we open during last close of application// we'll reopen them if no explicit files to open are provided in command linebool reopenLastFiles;QString lastFiles;QString activeFile;LC_GROUP("Startup");{reopenLastFiles = LC_GET_BOOL("OpenLastOpenedFiles");lastFiles = LC_GET_STR("LastOpenFilesList", "");activeFile = LC_GET_STR("LastOpenFilesActive", "");bool checkForNewVersion = LC_GET_BOOL("CheckForNewVersions", true);if (reopenLastFiles && fileList.isEmpty() && !lastFiles.isEmpty()) {foreach(const QString &filename, lastFiles.split(";")) {if (!filename.isEmpty() && QFileInfo::exists(filename))fileList << filename;}}bool files_loaded = false;for (QStringList::Iterator it = fileList.begin(); it != fileList.end(); ++it) {if (show_splash) {splash->showMessage(QObject::tr("Loading File %1..").arg(QDir::toNativeSeparators(*it)),Qt::AlignRight | Qt::AlignBottom, Qt::black);qApp->processEvents();}appWin.slotFileOpen(*it);files_loaded = true;}if (reopenLastFiles) {appWin.activateWindowWithFile(activeFile);}RS_DEBUG->print("main: loading files: OK");if (!files_loaded) {appWin.slotFileNewNew();}if (show_splash) {splash->finish(&appWin);splash.release();}if (checkForNewVersion) {appWin.checkForNewVersion();}}LC_GROUP_END();if (first_load)settings.setValue("Startup/FirstLoad", 0);RS_DEBUG->print("main: entering Qt event loop");QCoreApplication::processEvents();int return_code = app.exec();RS_DEBUG->print("main: exited Qt event loop");// Destroy the singletonQC_ApplicationWindow::getAppWindow().reset();return return_code;
}/*** Handles command line arguments that might not require a GUI.** @return list of files to load on startup.*/
QStringList handleArgs(int argc, char** argv, const QList<int>& argClean){RS_DEBUG->print("main: handling args..");QStringList ret;bool doexit = false;for (int i=1; i<argc; i++) {if(argClean.indexOf(i)>=0) continue;if (!QString(argv[i]).startsWith("-")){QString fname = QDir::toNativeSeparators(QFileInfo(QFile::decodeName(argv[i])).absoluteFilePath());ret.append(fname);}else if (QString(argv[i])=="--exit"){doexit = true;}}if (doexit) {exit(0);}RS_DEBUG->print("main: handling args: OK");return ret;
}QString LCReleaseLabel(){QString version{XSTR(LC_VERSION)};QString label;const std::map<QString, QString> labelMap = {{"rc", QObject::tr("Release Candidate")},{"beta", QObject::tr("BETA")},{"alpha", QObject::tr("ALPHA")}};for (const auto& [key, value]: labelMap) {if (version.contains(key, Qt::CaseInsensitive)) {label=value;break;}}return label;
}namespace {void restoreWindowGeometry(QC_ApplicationWindow& appWin, QSettings& settings){settings.beginGroup("Geometry");auto geometryB64 = settings.value("/WindowGeometry").toString().toUtf8();auto geometry = QByteArray::fromBase64(geometryB64, QByteArray::Base64Encoding);if (!geometry.isEmpty()) {appWin.restoreGeometry(geometry);} else {// fallbackint windowWidth = settings.value("WindowWidth", 1024).toInt();int windowHeight = settings.value("WindowHeight", 1024).toInt();int windowX = settings.value("WindowX", 32).toInt();int windowY = settings.value("WindowY", 32).toInt();appWin.resize(windowWidth, windowHeight);appWin.move(windowX, windowY);}settings.endGroup();}// Update Splash image to show "ALPHA", "BETA", and "Release Candidate"
QPixmap getSplashImage(const std::unique_ptr<QSplashScreen>& splash, const QString& label);
// Update Splash Screenvoid updateSplash(const std::unique_ptr<QSplashScreen>& splash){if (splash == nullptr)return;QString label = LCReleaseLabel();if (label.isEmpty())return;QPixmap splashImage = getSplashImage(splash, label);splash->setPixmap(splashImage);splash->setAttribute(Qt::WA_DeleteOnClose);splash->show();splash->showMessage(QObject::tr("Loading.."),Qt::AlignRight|Qt::AlignBottom, Qt::black);}// Update Splash image to show "ALPHA", "BETA", and "Release Candidate"QPixmap getSplashImage(const std::unique_ptr<QSplashScreen>& splash, const QString& label){if (splash == nullptr)return {};QPixmap pixmapSplash(":/main/splash_librecad.png");QPainter painter(&pixmapSplash);const double factorX = pixmapSplash.width()/542.;const double factorY = pixmapSplash.height()/337.;painter.setPen(QColor(255, 0, 0, 128));QRectF labelRect{QPointF{280.*factorX, 130.*factorY}, QPointF{480.*factorX, 170.*factorY}};QFont font;font.setPixelSize(int(labelRect.height()) - 2);painter.setFont(font);painter.drawText(labelRect,Qt::AlignRight, label);return pixmapSplash;}
}
二、代码介绍
- Qt 版本检查
QT_REQUIRE_VERSION(argc, argv, "5.2.1");
确保应用程序运行在 Qt 5.2.1 或更高版本上。
- 控制台模式检测
for (int i = 0; i < qMin(argc, 2); i++) {// 检查是否为 dxf2pdf/dxf2png 命令if (arg.compare("dxf2pdf") == 0) return console_dxf2pdf(...);if (arg.compare("dxf2png") == 0) return console_dxf2png(...);
}
检测是否以命令行工具模式运行,用于文件格式转换(如 PDF/PNG/SVG)。
- 应用程序初始化
LC_Application app(argc, argv);
QCoreApplication::setOrganizationName("LibreCAD");
// ... 其他应用程序设置 ...
RS_Settings::init(...);
-
创建 Qt 应用程序对象。
-
设置组织和应用程序的元数据。
-
初始化持久化设置系统。
- 命令行参数处理
QStringList fileList = handleArgs(argc, argv, argClean);
处理以下参数:
- 调试级别控制(-d/–debug)。
- 帮助信息输出(-h/–help)。
- 要打开的文件路径。
- 特殊命令(如 --exit)。
- 首次运行配置
if (first_load) {QG_DlgInitial di(nullptr);// 显示初始配置对话框
}
- 如果是第一次运行,显示初始设置对话框。
- 设置默认单位和其他基本偏好。
- 启动画面设置
auto splash = std::make_unique<QSplashScreen>();
updateSplash(splash); // 添加版本标签(如 ALPHA/BETA 等)
- 创建并自定义启动画面。
- 使用 getSplashImage() 添加版本标签。
- 资源初始化
RS_FONTLIST->init(); // 字体
RS_PATTERNLIST->init(); // 图案
RS_SYSTEM->loadTranslation(...); // 本地化
- 加载 CAD 所需的资源。
- 设置多语言支持。
- 主窗口创建
QC_ApplicationWindow& appWin = *QC_ApplicationWindow::getAppWindow();
restoreWindowGeometry(appWin, settings);
- 使用单例模式创建主应用程序窗口。
- 从设置中恢复窗口的几何状态。
- 文件处理
// 如果配置了,重新打开上次的文件
if (reopenLastFiles) fileList = lastFiles.split(";"); // 加载命令行指定的文件
for (QString file : fileList) appWin.slotFileOpen(file);
- 处理命令行指定的文件以及“重新打开上次文件”的功能。
- 如果没有指定文件,则创建新文档。
- 事件循环与清理
int return_code = app.exec();
// ... 清理 ...
return return_code;
- 启动 Qt 事件循环。
- 在退出时进行适当的清理。
- 在关闭时保存设置。
关键架构特性
- 单例模式:用于主应用程序窗口(QC_ApplicationWindow::getAppWindow())。
- 工厂模式:用于字体和图案资源的初始化。
- 命令分发:同时支持 GUI 和命令行模式。
- 持久化:使用 QSettings 保存窗口几何状态和偏好设置。
- 国际化:通过翻译文件支持多语言。
重要的辅助函数
- handleArgs():处理命令行参数。
- restoreWindowGeometry():从设置中恢复窗口状态。
- updateSplash():自定义启动画面,添加版本信息。
- LCReleaseLabel():确定版本类型(如 Alpha/Beta 等)。
总结
这段代码展示了一个成熟的 Qt 应用程序结构,具有清晰的职责分离:
- 初始化:应用程序和资源的初始化。
- 资源管理:字体、图案和本地化资源的加载。
- UI 设置:主窗口和启动画面的创建与配置。
- 命令行处理:支持命令行模式和文件加载。
通过良好的架构设计和模块化实现,LibreCAD 的主程序能够高效地处理多种运行场景,并为用户提供一致的使用体验。
相关文章:
【学写LibreCAD】1 LibreCAD主程序
一、源码 头文件: #ifndef MAIN_H #define MAIN_H#include<QStringList>#define STR(x) #x #define XSTR(x) STR(x)/*** brief handleArgs* param argc cli argument counter from main()* param argv cli arguments from main()* param argClean a list…...
CentOS7最小化安装中使用curl安装yum和wget
在 CentOS 7 最小化安装中,如果已经有curl工具,可以按照以下步骤使用它来安装yum和wget: 1. 备份原有的 yum 源配置文件 为了避免配置冲突或后续需要恢复,先备份原有的yum源配置文件。 mv /etc/yum.repos.d/CentOS-Base.repo /…...
【Linux】learning notes(3)make、copy、move、remove
文章目录 1、mkdir (make directory)2、rmdir (remove directory)3、rm(remove)4、>5、touch 新建文件6、mv(move)7、cp(copy) 1、mkdir (make…...
P10108 [GESP202312 六级] 闯关游戏
题目大意 如题 分析 设最佳通关方案为 { s 1 , s 2 , . . . , s k } \{s_1,s_2,...,s_k\} {s1,s2,...,sk},其中 s i s_i si 代表第 i i i 次到达的关卡( ≥ N \ge N ≥N 的不算)。 当 a k N − 1 a_kN-1 akN−1 时&#…...
Dubbo RPC 原理
一、Dubbo 简介 Apache Dubbo 是一款高性能、轻量级的开源 RPC 框架,支持服务治理、协议扩展、负载均衡、容错机制等核心功能,广泛应用于微服务架构。其核心目标是解决分布式服务之间的高效通信与服务治理问题。 二、Dubbo 架构设计 1. 核心组件 Prov…...
网络安全 机器学习算法 计算机网络安全机制
(一)网络操作系统 安全 网络操作系统安全是整个网络系统安全的基础。操作系统安全机制主要包括访问控制和隔离控制。 访问控制系统一般包括主体、客体和安全访问政策 访问控制类型: 自主访问控制强制访问控制 访问控制措施: 入…...
【Jenkins】一种灵活定义多个执行label节点的jenkinsfile写法
确定执行机器和自定义工作目录(忽略节点的workspace) pipeline{agent {node {label "XXXXX"customWorkspace "E:/workspace/"}}parameters {}options {}stages {}post {} }仅确定执行机器 pipeline{agent { label "XXXXX&quo…...
Web自动化之Selenium控制已经打开的浏览器(Chrome,Edge)
在使用selenium进行web自动化或爬虫的时候,经常会面临登录的情况,对于这种情况,我们可以利用Selenium控制已经打开的浏览器,从而避免每次都需要重新打开浏览器并进行登录的繁琐步骤。 目录 说明 启动浏览器 注意 --user-data-dir说明 代码设定 代码 改进代…...
【万字长文】开源之播对话白鲸开源CEO郭炜--乐观主义的开源精神走得更远
本文为白鲸开源科技CEO郭炜1小时深度访谈全记录 来源于:开源之播」Episode15:对话郭炜–乐观主义的开源精神走得更远 大家好,我是郭炜,开源圈的“郭大侠”。作为 Apache 基金会的成员,我曾参与并孵化了多个开源项目,如…...
Verilog 位运算符和逻辑运算符的使用
Verilog 位运算符和逻辑运算符的使用 目录 前言 一、逻辑运算符 二、位运算符 总结 前言 本文详细描述了Verilog 逻辑运算符和位运算符的使用,随着编程的熟练,有时候总是喜欢混用它们,虽然能实现同样的功能,但最好还是注意一下…...
压测报告:DeepSeek-R1-Distill-Qwen-32B模型性能评估
1. 实验背景 本实验旨在评估DeepSeek-R1-Distill-Qwen-32B模型在特定硬件配置下的性能表现。测试硬件为GPU服务器。实验主要关注模型在不同并发请求数下的峰值生成速度。 吞吐量(Throughput): 测试模型在单位时间内可以处理多少请求,通常以“每秒生成的令牌数(tokens/s)…...
【论文笔记】ClipSAM: CLIP and SAM collaboration for zero-shot anomaly segmentation
原文链接 摘要 近年来,CLIP 和 SAM 等基础模型在零样本异常分割 (ZSAS) 任务中展现出良好的性能。然而,无论是基于 CLIP 还是基于 SAM 的 ZSAS 方法,仍然存在不可忽视的关键缺陷:1) CLIP 主要关注不同输入之间的全局特征对齐&am…...
DeepSeek:面向效率与垂直领域的下一代大语言模型技术解析
本文将深入剖析DeepSeek模型的核心算法架构,揭示其在神经网络技术上的突破性创新,并与主流大模型进行全方位技术对比。文章涵盖模型设计理念、训练范式优化、应用场景差异等关键维度,为读者呈现大语言模型领域的最新发展图景。 一、DeepSeek…...
win32汇编环境,加速键的应用示例
;运行效果 ;win32汇编环境,加速键的应用示例 ;加速键,就是按某个键,开启某个功能。不用鼠标点来点去的东西。 ;直接抄进RadAsm可编译运行。重要部分加备注。 ;下面为asm文件 ;>>>>>>>>>>>>>>>>>>…...
【计算机网络】OSI模型、TCP/IP模型、路由器、集线器、交换机
一、计算机网络分层结构 计算机网络分层结构 指将计算机网络的功能划分为多个层次,每个层次都有其特定的功能和协议,并且层次之间通过接口进行通信。 分层设计的优势: 模块化:各层独立发展(如IPv4→IPv6,…...
[Web 安全] 反序列化漏洞 - 学习笔记
关注这个专栏的其他相关笔记:[Web 安全] Web 安全攻防 - 学习手册-CSDN博客 0x01:反序列化漏洞 — 漏洞介绍 反序列化漏洞是一种常见的安全漏洞,主要出现在应用程序将 序列化数据 重新转换为对象(即反序列化)的过程中…...
每日一题——字母异位词分组
字母异位词分组 1. 问题描述示例提示 2. 解题思路具体步骤 3. 代码实现4. 代码解析(1)排序法(2)哈希表存储(3)动态内存分配(4)释放内存1. HASH_FIND_STR 的作用2. 宏的定义4. 详细解…...
力扣 807. 保持城市天际线(Java实现)
题目分析 给定一个二维数组,行列长度相等,要保持四个方向仍一观察高度不变的情况下,适当添加建筑高度,问最大高度增量和。所谓四个方向高度不变的增量,其实就是arr[i][j]与同i行最大值同j列最大值之间的最小值的差&…...
【视频2 - 4】初识操作系统,Linux,虚拟机
📝前言说明: ●本专栏主要记录本人的基础算法学习以及LeetCode刷题记录,主要跟随B站博主灵茶山的视频进行学习,专栏中的每一篇文章对应B站博主灵茶山的一个视频 ●题目主要为B站视频内涉及的题目以及B站视频中提到的“课后作业”。…...
重启securecmd失败
重启securecmd失败 问题描述:KES集群部署工具中,节点管理里新增节点下一步报错无法检查securecmd端口进程情况,安装依赖包后再次下一步提示如下报错: 解决办法: [rootlocalhost cluster]# cd /home/kingbase/cluster…...
python学习四
python运算符与表达式 表达式: Python中的表达式是一种计算结果的代码片段。它可以包 含变量、运算符、常数和函数调用,用于执行各种数学、逻辑 和功能操作 算术运算符: 比较(关系)运算符: 赋值运算符: 逻辑运算符: 位运算符: 成员运算符: 身份运算符 <...
LeetCode 236.二叉树的最近公共祖先
题目: 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节…...
react 中,使用antd layout布局中的sider 做sider的展开和收起功能
一 话不多说,先展示效果: 展开时: 收起时: 二、实现代码如下 react 文件 import React, {useState} from react; import {Layout} from antd; import styles from "./index.module.less"; // 这个是样式文件&#…...
2025-02-26 学习记录--C/C++-C语言 整数格式说明符
合抱之木,生于毫末;九层之台,起于累土;千里之行,始于足下。💪🏻 C语言 整数格式说明符 【例如 】🎀 :在 C 语言中,%ld 是 printf 或 scanf 等格式化输入输出函…...
绕过过滤order by
一、常见绕过技术 1、注释符截断 利用注释符(如 --、#)截断后续查询,消除过滤逻辑的影响。 ORDER BY 1-- 若原查询为 SELECT * FROM table ORDER BY 用户输入,注入后可能忽略后续过滤逻辑。 2、大小写混淆/编码绕过 若过滤是大…...
NLP09-加强1-对比SVM
支持向量机(SVM) (一)导入 SVM 相关库 (二) 修改模型初始化 (三) 比较 朴素贝叶斯分类器 SVM分类器 支持向量机(SVM) 代码修改基于NLP09-朴素贝叶斯问句…...
STM32——HAL库开发笔记22(定时器3—呼吸灯实验)(参考来源:b站铁头山羊)
本文利用前几节所学知识来实现一个呼吸灯实验:两颗led灯交替呼吸。 一、STM32CubeMX配置 step1:配置调试接口 step2:配置定时器 定时器1位于APB2总线上,如上图所示。 step3:配置时基单元 按照下图配置 时钟来源配置…...
Vue3+TypeScript 封装一个好用的防抖节流自定义指令
一、前言:为什么需要防抖节流? 在前端开发中,高频触发的事件(如滚动、输入、点击等)容易导致性能问题。防抖(debounce) 和 节流(throttle) 是两种常用的优化手段&#x…...
2025年中国最新安防行业数字安全现状与未来趋势:内生安全成核心共识,从零基础到精通,收藏这篇就够了!
本次,我将为大家剖析由**安奇信发布的《中国安防行业数字安全建设与发展情况白皮书**》。该报告共_37_页,涵盖了众多重要信息和核心论点。若您希望深入了解,请参阅原报告,获取方法已在文档的最后部分提供。 报告核心内容 随着数字…...
项目管理的新理念主要有哪些
随着时代的发展和市场需求的变化,项目管理的理念也在不断地创新和演变。项目管理的新理念包括敏捷管理、精益管理、知识管理、变革管理、协作式管理和项目生命周期管理等。这些新理念不仅能够提高项目的管理效率,还能帮助团队在复杂的环境中更好地应对挑…...
Spring如何解决循环依赖?
一、Spring的三级缓存 关键就是提前暴露未完全创建完毕的Bean。 三级缓存来解决循环依赖: 一级缓存:用于存储完全初始化完成的单例Bean。 二级缓存:用于存储尚未完全初始化,但已实例化的Bean,用于提前暴露对象&#x…...
vscode java环境中文乱码的问题
先说我的结论: 由于我的系统是windows的,所以vscode使用的是默认gbk的编码进行的。 但是我的目的是全部都使用utf-8,因为我的程序始终是要去linux上去运行的,总不能在本地是好的,然后到服务器上就不行了吧,…...
十、大数据资源平台功能架构
一、大数据资源平台的功能架构图总体结构 大数据资源平台功能架构图 关键组件: 1.用户(顶行) 此部分标识与平台交互的各种利益相关者。 其中包括: 市领导 各部门分析师 区政府 外部组织 公民 开发人员 运营经理 2.功能模…...
OpenEuler学习笔记(三十五):搭建代码托管服务器
以下是主流的代码托管软件分类及推荐,涵盖自托管和云端方案,您可根据团队规模、功能需求及资源情况选择: 一、自托管代码托管平台(可私有部署) 1. GitLab 简介: 功能全面的 DevOps 平台,支持代码托管、C…...
使用Spring AI调用Ollama的DeepSeek模型实现结构化输出
在 Docker 环境中部署 Ollama 并使用 Spring AI 框架实现结构化输出,你可以按照以下步骤进行操作: 1. 部署 Ollama 模型 首先,需要在 Docker 中部署 Ollama 并下载 deepseek-r1:1.5b 模型。 1.1 准备部署文件 version: 3.8services:ollam…...
微信小程序调用火山方舟(字节跳动火山引擎)中的DeepSeek大模型
微信小程序的轻量化特性与DeepSeek大模型的AI能力结合,可快速构建智能问答、内容生成等场景化服务。通过火山方舟平台提供的标准化接口,开发者无需深入算法细节即可调用模型能力。 一、注册火山引擎账号,创建API Key和model(接入…...
threejs:射线拾取封装
射线拾取封装代码: import * as THREE from three; // 点击事件// 1.坐标转化(鼠标单击的屏幕坐标转标准设备坐标)// 2.射线计算(通过鼠标单击位置相机参数计算射线值)// 3.射线交叉计算// ObjectsArr是用来做射线拾取的对象数组,一个二维数组 export f…...
2024年国赛高教杯数学建模D题反潜航空深弹命中概率问题解题全过程文档及程序
2024年国赛高教杯数学建模 D题 反潜航空深弹命中概率问题 原题再现 应用深水炸弹(简称深弹)反潜,曾是二战时期反潜的重要手段,而随着现代军事技术的发展,鱼雷已成为现代反潜作战的主要武器。但是,在海峡或…...
爬虫框架与库
爬虫框架与库是用于网络数据抓取的核心工具,帮助开发者高效地从网页中提取结构化数据。 Requests:用于发送HTTP请求。 BeautifulSoup:用于解析HTML和XML。 Scrapy:强大的爬虫框架,适合大规模爬取。 Selenium&#…...
从电子管到量子计算:计算机技术的未来趋势
计算机发展的历史 自古以来人类就在不断地发明和改进计算工具,从结绳计数到算盘,计算尺,手摇计算机,直到1946年第一台电子计算机诞生,虽然电子计算机至今虽然只有短短的半个多世纪,但取得了惊人的发展吗,已经经历了五代的变革。计算机的发展和电子技术的发展密切相关,…...
LeetCode 15.三数之和
15.三数之和 给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k ,同时还满足 nums[i] nums[j] nums[k] 0 。请你返回所有和为 0 且不重复的三元组。 注意:答案中不可以包含重复的三元组…...
c#笔记-基础知识
目录 整数 浮点 decimal bool 引用 object string 值 枚举 关键字 enum 类型转换 隐式转换 编辑 显式转换 装箱,拆箱 常量 表达式 编辑 字符串的使用 比较字符串 字符格式化 编辑 截取字符串 字符串的分割 插入字符串 删除字符串 …...
el-table fixed滚动条被遮挡导致滚动条无法拖动
/* 设置默认高度-滚动条高度 */ .el-table__fixed { height: calc(100% - 16px) !important; } .el-table__fixed:before { height: 0px; }...
如何在个人电脑本地部署Deepseek大模型?Windows10 + Ollama+Deepseek+ChatBox
第一阶段:下载并安装Ollama 1,打开浏览器,进入 Ollama 官方网站 2,点击“Download”,在新页面选择“Windows”,再点击“Download for Windows”。 3,下载完成后,在文件名࿰…...
链表题()
1、环形列表 141 方法一、 卡节点数量,或者值范围的bug 方法二、 快慢指针(最实用)寻找环 /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode(int x) : val(x), next(NUL…...
ZIP64扩展和普通ZIP文件有什么区别?
ZIP64扩展是ZIP文件格式的一个扩展,旨在解决传统ZIP格式的限制,尤其是文件大小和数量的限制。以下是ZIP64扩展与普通ZIP文件的主要区别: 1. 文件大小限制 普通ZIP文件: 单个文件大小限制为 4GB(2^32字节)。…...
smolagents学习笔记系列(五)Tools-in-depth-guide
这篇文章锁定官网教程中的 Tools-in-depth-guide 章节,主要介绍了如何详细构造自己的Tools,在之前的博文 smolagents学习笔记系列(二)Agents - Guided tour 中我初步介绍了下如何将一个函数或一个类声明成 smolagents 的工具&…...
使用 Apache Dubbo 释放 DeepSeek R1 的全部潜力
作者:陈子康,Apache Dubbo Contributor 2025年1月20日,国产大模型公司深度求索(DeepSeek)正式发布了大语言模型 DeepSeek-R1,并同步开源其模型权重。通过大规模强化学习技术,DeepSeek-R1 显著提…...
【星云 Orbit-F4 开发板】03b. 按键玩法二:独立按键双击双击触发
【星云 Orbit-F4 开发板】03b. 按键玩法二:独立按键双击触发 引言 在嵌入式系统中,按键是常用的输入设备。通过检测按键的状态变化,可以实现多种功能,例如单击、双击、长按等。本文将详细介绍如何使用STM32F103的GPIO引脚检测独…...
ave-form.vue 组件中 如何将产品名称发送给后端 ?
如何将产品名称发送给后端。 在这段代码中,产品名称(productName)的处理和发送主要发生在 save() 方法中。让我逐步分析: 产品ID的选择: <w-form-selectv-model"form.productId"label"涉及产品&q…...