Qt/C++编写GB28181服务端工具/绿色版开箱即用/对标wvp-gb28181/实时画面预览/录像回放下载

article/2025/7/6 9:30:44

一、前言说明

使用过不少的gb28181服务端工具,绝大部分都是BS结构的,也就是直接在网页上运行,比如easynvr、liveqing等,也有个知名的开源国标项目叫wvp,总体感觉性能都不如意,理论上来说肯定不如直接CS结构的可执行文件来解码显示来的快,毕竟网页运行的东西必须完全依赖浏览器,而浏览器的性能在视频解码这块,比起本地解码来还是差很多的,而且很多时候无法用硬解码,还有就是在国产化大潮下,在国产芯片的电脑上运行网页程序,性能非常的不如意,也许若干年后会好很多。

目前市面上没有看到什么监控系统支持gb28181,支持国标的都是服务器上运行,而且基本上和流媒体服务绑定在一起,也就是国标流拉到后直接推给了流媒体服务,然后给出各种拉流地址让监控系统去拉流显示。这种目前是主流的做法。之前就做过onvif版本的视频监控系统,陆陆续续有不少人问过有没有gb28181版本的,当时看了下协议文档,觉得挺复杂的,就暂时搁置了很多年,最近一年刚好有个大项目需要,索性就直接研究摸透一下,用纯qt给整出来了,sip服务支持udp和tcp,拉流支持udp/tcp被动/tcp主动。发布后绿色版开箱即用。

二、效果图

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

三、相关地址

  1. 国内站点:https://gitee.com/feiyangqingyun
  2. 国际站点:https://github.com/feiyangqingyun
  3. 个人作品:https://blog.csdn.net/feiyangqingyun/article/details/97565652
  4. 文件地址:https://pan.baidu.com/s/1d7TH_GEYl5nOecuNlWJJ7g 提取码:01jf 文件名:bin_video_gb28181。

四、功能特点

  1. 支持设备注册、注销、心跳、校时、注册认证、注销认证等。
  2. 设备上线后可以手动获取设备状态、设备信息、配置信息、预置位信息等。
  3. 设备上线后自动获取设备通道信息,包括中文通道名称。识别到通道上线离线变化,会重新获取该设备的所有通道信息。
  4. 支持视频点播,可以分别点播主码流和子码流,内置rtp解包线程,解包后发给视频播放组件解码播放。
  5. 每个设备每个通道支持点播多个视频,通过ssrc区分,支持共用端口和不同端口收流。
  6. 支持对某个设备下面所有通道、某个通道、某个通道对应的某个流分别关闭。
  7. 支持录像文件查询和回放,回放控制支持暂停播放、继续播放、倍速播放、切换播放进度。
  8. 支持录像文件下载,支持倍速比如8倍速下载,可同时多线程批量下载。
  9. 回放和下载同时支持IPC和NVR,比如摄像头自带的SD存储卡录像文件回放,NVR上的硬盘录像文件回放。
  10. 支持云台控制,向上、向下、向左、向右、左上、右上、左下、右下方位移动,镜头放大缩小,光圈放大缩小,镜头聚焦放焦。
  11. 支持预置位信息的查询、调用、添加、修改、删除等操作。
  12. 自动目录订阅功能,通道上线下线都有对应的信号通知。
  13. 内置定时读取通道信息机制,以保证通道信息是最新的,比如有些NVR是不断更新的通道信息。
  14. 内置订阅警情和位置移动功能,订阅后各种警情事件比如运动目标检测报警、入侵检测报警、徘徊检测报警等自动上报。
  15. 支持语音对讲功能,可以直接在视频窗体的悬浮条上单击语音对讲按钮,再次单击关闭对讲,对讲期间悬浮条常驻显示。
  16. 支持设备布防撤防,布防后警情信息会主动上报。
  17. 国标服务同时支持udp和tcp方式,可选只监听一种或者两种都监听,tcp方式自动处理粘包问题。
  18. 国标拉流同时支持udp、tcp被动、tcp主动三种方式,每个通道都可以自由选择何种拉流方式。
  19. 内置拉流端口池,每次拉流从中取出一个,关闭流自动回收端口号,重复利用。
  20. 收流端口自动纠错,自动跳过被占用的端口,不会出现端口占用导致收流失败的情况。
  21. 支持三种取流方式自动检测离线重连,检测到离线后,自动重启点播拉流整个流程。
  22. 录像文件回放,上一个完成后自动切换到下一个继续回放,直到所有回放完成。支持高达8倍速回放。
  23. 视频播放自适应硬解码,极低资源占用,实时性极好,带悬浮条显示视频流信息,可以直接在悬浮条单击按钮保存录像文件到本地。
  24. 支持几千路国标消息交互并发,实时视频流支持64路同时显示,可以拓展更多路数。
  25. 支持阿里云等云服务器,可以分别设置内网监听地址和外网访问地址,一般云服务器上是监听地址用内网,对外访问用外网地址。
  26. 支持视频分发,也就是推流,视频通道打开后可以自动推流到流媒体服务器,其他需要的地方拉流即可,支持rtsp、rtmp、hls、webrtc等方式拉流。
  27. 实时预览和录像回放都支持推流,推流支持叠加文字和图片水印以及各种ffmpeg支持的滤镜效果,支持多个水印同时叠加。
  28. 同时支持gb28181-2011、gb28181-2016、gb28181-2022以及后续可能的所有协议版本。
  29. SIP解析和交互采用纯Qt底层代码实现,udp/tcp通信交互,祖传原创代码解析,不依赖任何第三方。
  30. 代码量少,gb28181交互部分共几千行代码,注释详细,接口友好,使用极其简单,提供非常详细的使用示例。
  31. 支持海康、大华、宇视、华为、天地伟业等所有国标设备,包括一些没有ssrc的设备。
  32. 支持所有Qt版本和编译器以及操作系统,包括但不限于win、linux、mac、android、嵌入式linux、树莓派香橙派、国产os等。

五、相关代码

#include "gb28181server.h"
#include "gb28181serverudp.h"
#include "gb28181servertcp.h"GB28181Server::GB28181Server(QObject *parent) : QObject(parent)
{//注册数据类型qRegisterMetaType<GB28181Event>("GB28181Event");qRegisterMetaType<GB28181Status>("GB28181Status");qRegisterMetaType<GB28181Preset>("GB28181Preset");qRegisterMetaType<GB28181Record>("GB28181Record");qRegisterMetaType<GB28181Channel>("GB28181Channel");qRegisterMetaType<QList<GB28181Preset> >("QList<GB28181Preset>");qRegisterMetaType<QList<GB28181Record> >("QList<GB28181Record>");qRegisterMetaType<QList<GB28181Channel> >("QList<GB28181Channel>");//国标服务udp通信gbUdpServer = new GB28181ServerUdp;gbUdpServer->setObjectName("gbUdpServer");connect(gbUdpServer, SIGNAL(sendData(QString, int, QString, QString)), this, SIGNAL(sendData(QString, int, QString, QString)));connect(gbUdpServer, SIGNAL(receiveData(QString, int, QString, QString)), this, SIGNAL(receiveData(QString, int, QString, QString)));connect(gbUdpServer, SIGNAL(receiveInfo(QString, int, QString, QString)), this, SIGNAL(receiveInfo(QString, int, QString, QString)));connect(gbUdpServer, SIGNAL(startVideo(QString, int, QString, QString, QString, int, int)), this, SIGNAL(startVideo(QString, int, QString, QString, QString, int, int)));connect(gbUdpServer, SIGNAL(startAudio(QString, int, QString, QString, QString, int, int)), this, SIGNAL(startAudio(QString, int, QString, QString, QString, int, int)));connect(gbUdpServer, SIGNAL(deviceChanged(QString, bool)), this, SIGNAL(deviceChanged(QString, bool)));connect(gbUdpServer, SIGNAL(channelChanged(QString, QList<GB28181Channel>)), this, SIGNAL(channelChanged(QString, QList<GB28181Channel>)));connect(gbUdpServer, SIGNAL(receiveEvent(GB28181Event)), this, SIGNAL(receiveEvent(GB28181Event)));connect(gbUdpServer, SIGNAL(receiveStatus(GB28181Status)), this, SIGNAL(receiveStatus(GB28181Status)));connect(gbUdpServer, SIGNAL(receivePosition(GB28181Position)), this, SIGNAL(receivePosition(GB28181Position)));connect(gbUdpServer, SIGNAL(receivePreset(QList<GB28181Preset>)), this, SIGNAL(receivePreset(QList<GB28181Preset>)));connect(gbUdpServer, SIGNAL(receiveRecord(QList<GB28181Record>)), this, SIGNAL(receiveRecord(QList<GB28181Record>)));//国标服务tcp通信gbTcpServer = new GB28181ServerTcp;gbTcpServer->setObjectName("gbTcpServer");connect(gbTcpServer, SIGNAL(sendData(QString, int, QString, QString)), this, SIGNAL(sendData(QString, int, QString, QString)));connect(gbTcpServer, SIGNAL(receiveData(QString, int, QString, QString)), this, SIGNAL(receiveData(QString, int, QString, QString)));connect(gbTcpServer, SIGNAL(receiveInfo(QString, int, QString, QString)), this, SIGNAL(receiveInfo(QString, int, QString, QString)));connect(gbTcpServer, SIGNAL(startVideo(QString, int, QString, QString, QString, int, int)), this, SIGNAL(startVideo(QString, int, QString, QString, QString, int, int)));connect(gbTcpServer, SIGNAL(startAudio(QString, int, QString, QString, QString, int, int)), this, SIGNAL(startAudio(QString, int, QString, QString, QString, int, int)));connect(gbTcpServer, SIGNAL(deviceChanged(QString, bool)), this, SIGNAL(deviceChanged(QString, bool)));connect(gbTcpServer, SIGNAL(channelChanged(QString, QList<GB28181Channel>)), this, SIGNAL(channelChanged(QString, QList<GB28181Channel>)));connect(gbTcpServer, SIGNAL(receiveEvent(GB28181Event)), this, SIGNAL(receiveEvent(GB28181Event)));connect(gbTcpServer, SIGNAL(receiveStatus(GB28181Status)), this, SIGNAL(receiveStatus(GB28181Status)));connect(gbTcpServer, SIGNAL(receivePosition(GB28181Position)), this, SIGNAL(receivePosition(GB28181Position)));connect(gbTcpServer, SIGNAL(receivePreset(QList<GB28181Preset>)), this, SIGNAL(receivePreset(QList<GB28181Preset>)));connect(gbTcpServer, SIGNAL(receiveRecord(QList<GB28181Record>)), this, SIGNAL(receiveRecord(QList<GB28181Record>)));//国标服务udp线程/启动线程后就启动服务/线程关闭则释放对象gbUdpThread = new QThread;gbUdpServer->moveToThread(gbUdpThread);connect(gbUdpThread, SIGNAL(started()), gbUdpServer, SLOT(start()));connect(gbUdpThread, SIGNAL(finished()), gbUdpServer, SLOT(deleteLater()));//国标服务tcp线程/启动线程后就启动服务/线程关闭则释放对象gbTcpThread = new QThread;gbTcpServer->moveToThread(gbTcpThread);connect(gbTcpThread, SIGNAL(started()), gbTcpServer, SLOT(start()));connect(gbTcpThread, SIGNAL(finished()), gbTcpServer, SLOT(deleteLater()));
}GB28181Server::~GB28181Server()
{this->stop();
}void GB28181Server::setServerPara(GB28181ServerPara serverPara)
{gbUdpServer->setServerPara(serverPara);gbTcpServer->setServerPara(serverPara);
}QStringList GB28181Server::getDeviceIps()
{QStringList ips;QStringList ips1 = gbUdpServer->getDeviceIps();QStringList ips2 = gbTcpServer->getDeviceIps();foreach (QString ip, ips1) {if (!ips.contains(ip)) {ips << ip;}}foreach (QString ip, ips2) {if (!ips.contains(ip)) {ips << ip;}}return ips;
}QList<GB28181Device> GB28181Server::getDevices()
{QList<GB28181Device> devices;QList<GB28181Device> devices1 = gbUdpServer->getDevices();QList<GB28181Device> devices2 = gbTcpServer->getDevices();foreach (GB28181Device device, devices1) {if (!devices.contains(device)) {devices << device;}}foreach (GB28181Device device, devices2) {if (!devices.contains(device)) {devices << device;}}return devices;
}void GB28181Server::start(ListenMode listenMode)
{if (listenMode == ListenMode_Udp) {gbUdpThread->start();} else if (listenMode == ListenMode_Tcp) {gbTcpThread->start();} else {gbUdpThread->start();gbTcpThread->start();}
}void GB28181Server::stop()
{gbUdpThread->quit();gbTcpThread->quit();
}void GB28181Server::query(const QString &deviceId, const QString &cmdType)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->query(deviceId, cmdType);} else {gbTcpServer->query(deviceId, cmdType);}
}void GB28181Server::queryConfig(const QString &deviceId)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->queryConfig(deviceId);} else {gbTcpServer->queryConfig(deviceId);}
}void GB28181Server::queryPreset(const QString &deviceId, const QString &channelId)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->queryPreset(deviceId, channelId);} else {gbTcpServer->queryPreset(deviceId, channelId);}
}void GB28181Server::queryRecord(const QString &deviceId, const QString &channelId, const QDateTime &dateStart, const QDateTime &dateEnd)
{QString startTime = dateStart.toString("yyyy-MM-ddThh:mm:ss");QString endTime = dateEnd.toString("yyyy-MM-ddThh:mm:ss");this->queryRecord(deviceId, channelId, startTime, endTime);
}void GB28181Server::queryRecord(const QString &deviceId, const QString &channelId, const QString &startTime, const QString &endTime)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->queryRecord(deviceId, channelId, startTime, endTime);} else {gbTcpServer->queryRecord(deviceId, channelId, startTime, endTime);}
}void GB28181Server::subscribe(const QString &deviceId, const QString &channelId)
{QDateTime now = QDateTime::currentDateTime();QString startTime = now.toString("yyyy-MM-ddT00:00:00");QString endTime = now.toString("yyyy-MM-ddT23:59:59");this->subscribe(deviceId, channelId, "Alarm", 3600, startTime, endTime);
}void GB28181Server::subscribe(const QString &deviceId, const QString &channelId, const QString &cmdType, int expires, const QString &startTime, const QString &endTime)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->subscribe(deviceId, channelId, cmdType, expires, startTime, endTime);} else {gbTcpServer->subscribe(deviceId, channelId, cmdType, expires, startTime, endTime);}
}void GB28181Server::broadcast(const QString &deviceId, const QString &channelId, int audioPort)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->broadcast(deviceId, channelId, audioPort);} else {gbTcpServer->broadcast(deviceId, channelId, audioPort);}
}void GB28181Server::ptz(const QString &deviceId, const QString &channelId, PtzType type, quint8 speed)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->ptz(deviceId, channelId, type, speed);} else {gbTcpServer->ptz(deviceId, channelId, type, speed);}
}void GB28181Server::guard(const QString &deviceId, const QString &channelId, bool arming)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->guard(deviceId, channelId, arming);} else {gbTcpServer->guard(deviceId, channelId, arming);}
}QString GB28181Server::invite(const QString &deviceId, const QString &channelId, int videoPort, const QString &startTime, const QString &endTime, int speed, int profile, TransmitMode mode, bool playback)
{if (gbUdpServer->exist(deviceId)) {return gbUdpServer->invite(deviceId, channelId, videoPort, startTime, endTime, speed, profile, mode, playback);} else {return gbTcpServer->invite(deviceId, channelId, videoPort, startTime, endTime, speed, profile, mode, playback);}
}void GB28181Server::bye(const QString &deviceId, const QString &channelId, const QString &ssrc)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->bye(deviceId, channelId, ssrc);} else {gbTcpServer->bye(deviceId, channelId, ssrc);}
}void GB28181Server::playControl(const QString &deviceId, const QString &channelId, PlayControl control, float value, const QString &ssrc)
{if (gbUdpServer->exist(deviceId)) {gbUdpServer->playControl(deviceId, channelId, control, value, ssrc);} else {gbTcpServer->playControl(deviceId, channelId, control, value, ssrc);}
}

http://www.hkcw.cn/article/CdtYQvafEi.shtml

相关文章

ProxyPin抓APK数据包

ProxyPin抓APK数据包&#xff1a; 下载地址&#xff1a;https://github.com/wanghongenpin/proxypin 环境配置&#xff1a; 夜神模拟器&#xff0c;开心消消乐&#xff0c;ProxyPin.apk&#xff0c;ProxyPin.exe 使用步骤&#xff1a; ProxyPin.apk安装https证书&#xff0…

空间智能重塑未来治理

当上海复兴岛的战略空间更新与科创策源功能在时空创新实验基地实现耦合共生&#xff0c;当武汉马拉松的智慧安防系统通过"实景三维城市智眼"构筑起全域智能防线&#xff0c;我们正见证着空间智能技术重构城市治理范式的革命性变革。这场以数据为血脉、算法为神经、模…

FastAPI安全认证:从密码到令牌的魔法之旅

title: FastAPI安全认证:从密码到令牌的魔法之旅 date: 2025/06/02 13:24:43 updated: 2025/06/02 13:24:43 author: cmdragon excerpt: 在FastAPI中实现OAuth2密码流程的认证机制。通过创建令牌端点,用户可以使用用户名和密码获取JWT访问令牌。代码示例展示了如何使用Cry…

Java Script函数

1.认识JS函数 1.1程序中的foo、bar、baz 在学习编程中&#xff0c;你可能会经常看到foo、bar、baz这些名词 它们通常被用来作为函数、变量、文件的名称 目前已经编程了计算机编程的术语一部分 但是它们本身并没有特别的用途和意义 常常被称之为“伪变量”&#xff08;metasynt…

吴恩达MCP课程(5):mcp_chatbot_prompt_resource.py

前提条件&#xff1a; 1、吴恩达MCP课程&#xff08;5&#xff09;&#xff1a;research_server_prompt_resource.py 2、server_config_prompt_resource.json文件 {"mcpServers": {"filesystem": {"command": "npx","args"…

性能测试的概念和场景设计

一、什么是性能 性能的定义&#xff1a;性能是指系统或设备在执行特定任务时的时间效率和资源利用情况。可以从两个主要维度来理解&#xff1a; 时间维度&#xff1a; a、响应时间&#xff1a;指系统从接收用户请求到返回结果所需的时间。 b、吞吐量&#xff1a;指单位时间内…

【Java学习笔记】异常

异常&#xff08;Exception&#xff09; 一、基本介绍 在 Java 程序中&#xff0c;将运行中发生的不正常情况称为 “异常”&#xff0c;开发过程中的语法错误和运行时发生的异常情况是不一样的。 二、异常的分类 1. Error&#xff08;错误&#xff09;&#xff1a;Java 虚拟…

50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Form Wave(表单label波动效果)

&#x1f4c5; 我们继续 50 个小项目挑战&#xff01;—— FormWave组件 仓库地址&#xff1a;https://github.com/SunACong/50-vue-projects 项目预览地址&#xff1a;https://50-vue-projects.vercel.app/ &#x1f3af; 组件目标 构建一个美观、动态的登录表单&#xff0…

数据库系统概论(十六)数据库安全性(安全标准,控制,视图机制,审计与数据加密)

数据库系统概论&#xff08;十六&#xff09;数据库安全性 前言一、数据库安全性1. 什么是数据库安全性&#xff1f;2. 为何会存在安全问题&#xff1f; 二、安全标准的发展1. 早期的“开拓者”&#xff1a;TCSEC标准2. 走向国际统一&#xff1a;CC标准3. TCSEC和CC标准有什么不…

显示即战略:铁电液晶如何成为 “数字中国” 的 “像素基石”?

一、显示技术&#xff1a;数字时代的核心战略支点 &#xff08;一&#xff09;从 “视觉窗口” 到 “战略基础设施” 在数字经济蓬勃发展的当下&#xff0c;显示技术早已超越了单纯的 “视觉呈现” 范畴&#xff0c;成为连接人与数字世界的关键接口。从智能手机、平板电脑到车…

⚡️ Linux grep 命令参数详解

⚡️ Linux grep 用法及参数详解 &#x1f4d8; 1. grep 简介 grep 是 Linux/Unix 系统中用于文本搜索的命令&#xff0c;其全称为 Global Regular Expression Print&#xff0c;意为全局正则表达式打印器。 它根据给定的 模式&#xff08;pattern&#xff09; 对文件或标准…

Rust 变量与可变性

文章目录 变量与可变性常量遮蔽&#xff08;Shadowing&#xff09; 变量与可变性 Rust中变量默认是不可变的&#xff0c;这是 Rust 鼓励你编写更安全、易于并发代码的众多方式之一。不过&#xff0c;你仍然可以选择让变量可变。让我们来探讨 Rust 为什么鼓励你优先使用不可变性…

YOLOV7改进之融合深浅下采样模块(DSD Module)和轻量特征融合模块(LFI Module)

目录 一、研究背景​ 二. 核心创新点​ ​2.1 避免高MAC操作​ ​2.2 DSDM-LFIM主干网络​ 2.3 P2小目标检测分支​ ​3. 代码复现指南​ 环境配置 关键修改点 ​4. 实验结果对比​ 4.1 VisDrone数据集性能 4.2 边缘设备部署 4.3 检测效果可视化 ​5. 应用场景​ …

(Python网络爬虫);抓取B站404页面小漫画

目录 一. 分析网页 二. 准备工作 三. 实现爬虫 1. 抓取工作 2. 分析工作 3. 拼接主函数&运行结果 四. 完整代码清单 1.多线程版本spider.py&#xff1a; 2.异步版本async_spider.py&#xff1a; 经常逛B站的同志们可能知道&#xff0c;B站的404页面做得别具匠心&…

Qt OpenGL 3D 编程入门

Qt 提供了强大的 OpenGL 集成功能&#xff0c;使得在 Qt 应用中实现 3D 图形变得更加简单。以下是使用 Qt 进行 OpenGL 3D 编程的基础知识。 1. 环境配置 创建 Qt 项目 新建 Qt Widgets Application 项目 在 .pro 文件中添加 OpenGL 模块&#xff1a; qmake QT co…

山东警方:10岁失联男童在河道溺亡 搜救行动告终

6月2日,山东滕州市公安局发布警情通报。5月31日22时35分许,警方接到孔某某报警,称其10岁的外孙赵某某于当天17时许离家后失联。接警后,公安机关迅速调阅监控、走访群众,并联合当地政府和社会救援力量,使用搜救警犬和无人机持续开展搜寻工作。6月2日15时许,在邻村一河道内…

韩国大选目前选情如何 李在明领跑选战

韩国第21届总统选举定于6月3日举行,主要候选人仍在抓紧最后机会展开竞选活动,争取更多选票。此次大选在前总统尹锡悦被弹劾后举行,共有7名候选人登记参选,但其中两人已宣布退选,最终有5名候选人参加角逐。这5名候选人分别是共同民主党候选人李在明、国民力量党候选人金文洙…

【深度学习新浪潮】以Dify为例的大模型平台的对比分析

我们从核心功能、适用群体、易用性、可扩展性和安全性五个维度展开对比分析: 一、核心功能对比 平台核心功能多模型支持插件与工具链Dify低代码开发、RAG增强、Agent自律执行、企业级安全支持GPT-4/5、Claude、Llama3、Gemini及开源模型(如Qwen-VL-72B),支持混合模型组合可…

【JAVA】注解+元注解+自定义注解(万字详解)

&#x1f4da;博客主页&#xff1a;代码探秘者 ✨专栏&#xff1a;《JavaSe》 其他更新ing… ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更新的动力❤️ &#x1f64f;作者水平有限&#xff0c;欢迎各位大佬指点&…

unity随机生成未知符号教程

目录 前言方法1方法2脚本后言示例代码 前言 在某些游戏中&#xff0c;有一些让人感到意味不明的未知符号&#xff0c;例如在游戏《巴别塔圣歌》中&#xff0c;就有这样一些能让人在初次就看不懂的未知符号。 或者在其他时候&#xff0c;这些未知符号如果跟粒子系统结合在一起的…