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

Triton + RISC-V忱

. GIF文件结构相比于 WAV 文件的简单粗暴GIF 的结构要精密得多因为它天生是为了网络传输而设计的包含了压缩机制。当我们用二进制视角观察 GIF 时它是由一个个 数据块Block 组成的数据块 (Block Name) 中文名称 字节数 (Bytes) 作用与核心逻辑Header 头标识 6 一般为 GIF89a当然也有GIF87a,用于声明这是一个GIF文件采用xx标准。Logical Screen Descriptor 逻辑屏幕描述符 7 画布设定定义图像总宽高、背景色索引及是否使用全局调色板。Global Color Table 全局颜色表 3 × N 颜料盘存储 RGB 颜色如 00 FF 00N 为颜色数最大256。 默认使用RGB颜色时每个颜色均采用3个字节存储Application Extension 应用程序扩展 19 (通常) 最常用的是 Netscape 扩展用于“循环播放次数”。Graphic Control Extension 图形控制扩展 8 播放控制定义每一帧的延迟时间动画快慢和透明色。Image Descriptor 图像描述符 10 帧属性定义当前这一帧在画布上的位置x, y和尺寸。Image Data 图像数据 可变 用处如其名Trailer 结束标识 1 终点固定为 0x3B (分号)标志文件彻底结束。其中最重要的就是图像数据了其他的块用于规定这些图像数据应当如何呈现到我们眼中或是告知文件的开始结束因此对于我们来说其他块基本上都有固定模板只有图像数据需要我们自己定义。2.LZW-GIF强制使用的图像压缩算法搞定了GIF的文件结构接下来就需要我们解决另一个拦路虎LZW——一个经典的无损图像压缩算法为了解决他我们可以从他的原理入手。LZW通过为复杂数据构建简单索引来减少存储的数据量这一点是朴素的哈希算法当然这一算法的发明者通过一套特殊的规则使得其他人可以直接通过索引数据反推出复杂数据而在GIF中则是GIF发明公司将他所规定的规则写好编写GIF的人根据这一套规则构建数据然后其他人直接使用套用了这一套规则的解码器解码便能将数据还原成原来的样子。GIF的解码器又是如何读取数据的呢解码器初始时一次性读取9位数据然后从字典中添加这一对应关系根据GIF规范一旦字典里的条目达到 512 个解码器就会自动将读取位宽从 9 位增加到 10 位。如果我们直接存放数据那么结果就是数据读取错位解码出来的内容就会与我们想象中的不一样如果我们想要让他的数据读取正常通常做法就是我们构建一个同步状态机即模拟GIF解码器读取数据的过程写入数据构建一个字典以相同的标准增加写入位宽从而让解码器读取时能正确读取。但是这一过程看着就十分繁琐能不能用一个简单的方法来让解码器正常读取数据呢3.解决方案GIF 协议中有一个特殊的指令叫 Clear Code清除代码值为256。它的作用是告诉解码器“嘿把之前的字典都忘了吧我们重新开始。”利用这一点我们可以在代码中采用了一种偷鸡的策略我们不尝试去寻找复杂的重复模式。我们每写入一小段像素例如 125 个就立刻发送一个 Clear Code。这强制让 LZW 字典始终处于“初始状态”。在初始状态下LZW 的编码就等同于直接输出像素的颜色索引值。现在让我们来实现他4. 构建3d立方体在上一篇关于 3D 本质的文章中我们推导出了两个核心公式。在这个程序中我们将直接把它们转化为 C 代码。旋转公式为了让立方体动起来我们需要每一帧都改变顶点的坐标。这里使用旋转矩阵的简化版Point3D rotate(Point3D p, float angle) {// 绕 Y 轴旋转float nx p.x * cos(angle) - p.z * sin(angle);float nz p.x * sin(angle) p.z * cos(angle);// 绕 X 轴微调旋转让旋转看起来更立体float ny p.y * cos(angle * 0.8f) - nz * sin(angle * 0.8f);nz p.y * sin(angle * 0.8f) nz * cos(angle * 0.8f);return {nx, ny, nz};}投影公式透视如何把 3D 坐标变成屏幕上的像素点记得那个核心法则吗“近大远小本质就是除以 Z”。pair project(Point3D p, int W, int H) {float fov 160.0f; // 视野系数float viewer_dist 4.0f; // 眼睛离物体的距离// 核心逻辑除以 (z dist)float factor fov / (viewer_dist p.z);return { (int)(p.x * factor W / 2), (int)(p.y * factor H / 2) };}有了这两个函数我们就能在内存里的一个二维数组vector pixels上画线了。手写 GIF 编码器我们实现了一个极简的编码器struct GifBitStream。它的工作是把像素点的颜色索引0或1打包成变长的二进制码流。// GIF 的数据存储不仅是字节还需要处理“位操作”// 比如写入一个 9-bit 的代码可能跨越两个字节struct GifBitStream {vector byteData;u32 bitBuffer 0;int bitCount 0;void writeCode(u32 code, int size) {// 将数据移位并存入缓冲区...// 凑够8位就写入 byteData}// ...};特别说明绕过LZW的问题生成的GIF没有压缩体积较为大4. 完整代码下面是完整的 C 代码计算旋转 3D 点 - 投影成 2D 点。绘图在内存的黑板上画线Bresenham 直线算法。编码将内存的黑板按照 GIF 协议压缩并写入文件。(代码较长建议直接复制编译运行感受生成的快感)#includeusing namespace std;#define u8 uint8_t#define u16 uint16_t#define u32 uint32_tstruct Point3D {float x, y, z;};struct Edge {int u, v;};Point3D rotate(Point3D p, float angle) {float nx p.x * cos(angle) - p.z * sin(angle);float nz p.x * sin(angle) p.z * cos(angle);float ny p.y * cos(angle * 0.8f) - nz * sin(angle * 0.8f);nz p.y * sin(angle * 0.8f) nz * cos(angle * 0.8f);return {nx, ny, nz};}pair project(Point3D p, int W, int H) {float fov 160.0f;float viewer_dist 4.0f;float factor fov / (viewer_dist p.z);return { (int)(p.x * factor W / 2), (int)(p.y * factor H / 2) };}inline void drawLine(vector buffer, int W, int H, int x0, int y0, int x1, int y1) {int dx abs(x1 - x0), sx x0 x1 ? 1 : -1;int dy -abs(y1 - y0), sy y0 y1 ? 1 : -1;int err dx dy;while (true) {if (x0 0 x0 W y0 0 y0 H) buffer[y0 * W x0] 1;if (x0 x1 y0 y1) break;int e2 2 * err;if (e2 dy) { err dy; x0 sx; }if (e2 dx) { err dx; y0 sy; }}}// --- GIF 二进制协议部分 ---struct GifBitStream {vector byteData;u32 bitBuffer 0;int bitCount 0;// 写入指定位宽的代码inline void writeCode(u32 code, int size) {bitBuffer | (code bitCount);bitCount size;while (bitCount 8) {byteData.push_back(bitBuffer 0xFF);bitBuffer 8;bitCount - 8;}}inline void flush(ofstream f) {if (bitCount 0) byteData.push_back(bitBuffer 0xFF);// GIF 规定数据必须切成每块最大 255 字节的小块for (size_t i 0; i byteData.size(); i 255) {u8 blockSize (u8)min((size_t)255, byteData.size() - i);f.put(blockSize);f.write((char*)byteData[i], blockSize);}f.put(0); // 块结束}};inline void writeWord(ofstream f, u16 v) {f.put(v 0xFF);f.put((v 8) 0xFF);}inline void writeGifFrame(ofstream f, const vector pixels, int W, int H) {// 1. 图形控制扩展 (帧间隔)f.put(0x21);f.put(0xF9);f.put(0x04);f.put(0x09); // 属性还原背景不使用透明writeWord(f, 4); // 延迟 40ms (1/25 FPS)f.put(0);f.put(0);// 2. 图像描述符f.put(0x2C);// 偏移writeWord(f, 0);writeWord(f, 0);// 宽高writeWord(f, W);writeWord(f, H);f.put(0x00);// 3. 数据f.put(0x08); // 8位色GifBitStream stream;const int ClearCode 256; //清空指令const int EOICode 257;stream.writeCode(ClearCode, 9); // 清空解码器字典int pixCount 0;for (u8 p : pixels) {stream.writeCode(p, 9);pixCount;//每 125 个像素重置一次字典保证位宽不变不会提前读取到下一个字节if (pixCount 125) {stream.writeCode(ClearCode, 9);pixCount 0;}}stream.writeCode(EOICode, 9); // 结束stream.flush(f);}signed main(int argc,char* argv[]){const int W 200, H 200;ofstream f(cube_perfect.gif, ios::binary); //以二进制方式写入GIF文件// [Header]f GIF89a;//89a 标准// [Logical Screen Descriptor]writeWord(f, W); writeWord(f, H);f.put(0xF7); // 开启全局调色板 (256色)f.put(0);f.put(0);// [Global Color Table]全局调色板// 0: 背景黑f.put(0);f.put(0);f.put(0);// 1: 极客绿f.put(0);f.put(255);f.put(0);//只用到两种颜色其余填充黑色for(int i 2; i 256; i){f.put(0); f.put(0); f.put(0);}// [Netscape Loop] 循环动画扩展f.put(0x21); // Netscape块标识f.put(0xFF); // 扩展类型标识f.put(0x0B); // 信息长度f NETSCAPE2.0; //应用程序信息f.put(0x03); // 数据长度到结束符前f.put(0x01); //索引writeWord(f, 0); //无限循环不停止f.put(0); //结束符// 3D 立方体点数据vector verts {{-1,-1,1}, {1,-1,1}, {1,1,1}, {-1,1,1},{-1,-1,-1}, {1,-1,-1}, {1,1,-1}, {-1,1,-1}};vector edges {{0,1},{1,2},{2,3},{3,0}, {4,5},{5,6},{6,7},{7,4}, {0,4},{1,5},{2,6},{3,7}};cout Encoding 3D Cube to GIF... endl;for (int i 0; i 60; i) { // 60帧动画vector pixels(W * H, 0); // 黑色背景float angle i * 0.12f; // 每1/60s旋转角度vector p2d;for (auto v : verts)p2d.push_back(project(rotate(v, angle), W, H)); // 3D-2D投影for (auto e : edges)drawLine(pixels, W, H, p2d[e.u].first, p2d[e.u].second, p2d[e.v].first, p2d[e.v].second); // 画边writeGifFrame(f, pixels, W, H); // 写入帧数据cout .;}f.put(0x3B); // 文件结束符f.close(); //关闭cout \nSuccess! Open cube_perfect.gif in Chrome/Edge. endl;return 0;}运行这段代码你会惊讶地发现目录下多了一个 cube_perfect.gif。用浏览器打开它一个绿色的线框立方体正在黑色的背景中流畅地旋转。5. 打通认知的“任督二脉”回顾这个系列的三篇文章我们其实只做了一件事祛魅Demystification。WAV 篇我们发现声音文件只是记录振幅的二进制队列没有任何魔法。3D 篇我们发现那些酷炫的 3D 游戏底层只是初中几何的“相似三角形”运算。GIF 篇本文我们将数学运算的结果3D按照文件协议二进制封装成了人类可见的动画。这就是计算机科学最迷人的地方。无论是生成一段 440Hz 的正弦波还是渲染《黑神话悟空》中复杂的场景其本质都是一样的Input数据 Rules算法/格式 Output数字世界掌喊衙咐

相关文章:

Triton + RISC-V忱

. GIF文件结构 相比于 WAV 文件的简单粗暴,GIF 的结构要精密得多,因为它天生是为了网络传输而设计的(包含了压缩机制)。 当我们用二进制视角观察 GIF 时,它是由一个个 数据块(Block) 组成的&…...

嵌入式按钮事件处理库:多类型去抖与状态机驱动设计

1. 项目概述 r89m Buttons 是一个面向嵌入式系统的轻量级、可移植按钮事件处理库,专为统一管理多种物理形态与电气特性的按钮输入而设计。其核心目标并非仅实现“按下/释放”电平检测,而是构建一套 事件驱动的抽象层 ,将底层硬件差异&…...

CCC3.0数字钥匙系统架构解析:从蓝牙OOB配对到多设备互操作性

1. 从机械钥匙到数字钥匙的技术演进 记得十年前我第一次买车时,销售递给我的是一把沉甸甸的机械钥匙,上面还挂着一个印着品牌logo的钥匙扣。那时候根本想不到,短短几年后我们就能用手机解锁汽车。这种变化背后,是CCC(C…...

MATLAB+CPLEX仿真平台下的微网虚拟电厂日前优化调度模型:融合电动汽车出行及充放电规律...

MATLAB代码:含多种需求响应及电动汽车的微网/虚拟电厂日前优化调度 关键词:需求响应 空调负荷 电动汽车 微网优化调度 虚拟电厂调度 仿真平台:MATLABCPLEX 主要内容:代码主要做的是一个微网/虚拟电厂的日前优化调度模型&#…...

STM32duino VL53L0X驱动深度解析:ToF传感器嵌入式实践指南

1. STM32duino VL53L0X 库深度解析:面向嵌入式工程师的ToF传感器驱动实践指南VL53L0X 是意法半导体(STMicroelectronics)推出的第二代飞行时间(Time-of-Flight, ToF)激光测距传感器,采用940nm不可见红外VCS…...

新手入门RTOS,别再纠结了!从RT-Thread和FreeRTOS的实战项目选择说起

新手入门RTOS:从实战项目看RT-Thread与FreeRTOS的选择策略 第一次接触实时操作系统(RTOS)时,面对众多选择往往会感到迷茫。作为嵌入式开发领域的核心技术之一,RTOS的选择直接影响着项目的开发效率和最终性能表现。在众…...

Vue中手动取消watch监听的最佳实践与实现原理

1. 为什么需要手动取消watch监听 在Vue开发中,watch监听器是我们常用的响应式工具之一。它能够监听数据变化并执行相应的回调函数。但很多开发者可能没有意识到,不当管理watch监听器可能会导致内存泄漏和性能问题。 想象一下这样的场景:你在一…...

BigEarthNet-MM数据集太大跑不动?教你用TFRecord分片和增量处理加速实验

BigEarthNet-MM数据集优化处理实战:分片技术与增量加载全解析 当你的GPU风扇开始发出直升机般的轰鸣,而TensorFlow进度条像树懒散步一样缓慢时——这可能是BigEarthNet-MM数据集在提醒你:传统的全量加载方式已经不适合这个时代了。本文将带你…...

数据摄取构建模块简介(预览版)(一)弛

一、语言特性:Java 26 与模式匹配进化 1.1 Java 26 语言级别支持 IDEA 2026.1 EAP 最引人注目的变化之一,就是新增 Java 26 语言级别支持。这意味着开发者可以提前体验和测试即将在 JDK 26 中正式发布的语言特性。 其中最重要的变化是对 JEP 530 的全面支…...

教育部:加快普及中小学生人工智能教育政策汇总

教育部:加快普及中小学生人工智能教育政策汇总 基本信息 发布时间:2026-04-10(最新政策)政策文件:《"人工智能教育"行动计划》发文机构:教育部、国家发展改革委、工业和信息化部、科技部、国家…...

从“单细胞”到“多细胞”:MetaGPT、AutoGen、AgentVerse如何重塑AI应用开发范式?

从“单细胞”到“多细胞”:MetaGPT、AutoGen、AgentVerse如何重塑AI应用开发范式? 想象一下,当你对AI说"开发一个电商网站"时,不再只是得到零散的代码片段,而是一个完整的数字团队自动分工协作:产…...

Adafruit Protomatter:HUB75 LED矩阵的裸机GPIO驱动原理与实践

1. Adafruit Protomatter 库深度技术解析:面向 HUB75 RGB LED 矩阵的裸机 GPIO 驱动框架 1.1 核心定位与工程目标 Adafruit Protomatter 是一个专为驱动 HUB75 接口 RGB LED 矩阵而设计的轻量级、高可移植性底层库。其核心设计哲学并非追求极致性能,而是…...

保姆级教程:在Jetson Orin上从零搭建PyTorch+TensorFlow环境(含torchvision源码编译避坑)

保姆级教程:在Jetson Orin上从零搭建PyTorchTensorFlow环境(含torchvision源码编译避坑) NVIDIA Jetson Orin作为当前边缘计算领域的旗舰平台,其ARM架构下的深度学习环境配置一直是开发者的痛点。本文将手把手带你完成从系统准备到…...

字符串拼接用“+”还是 StringBuilder?别再凭感觉写了品

前言 Kubernetes 本身并不复杂,是我们把它搞复杂的。无论是刻意为之还是那种虽然出于好意却将优雅的原语堆砌成 鲁布戈德堡机械 的狂热。平台最初提供的 ReplicaSets、Services、ConfigMaps,这些基础组件简单直接,甚至显得有些枯燥。但后来我…...

浅谈MIKE前处理中投影坐标处理问题

MIKE 中投影坐标一直是个问题,尤其对 2d 里的科氏力影响很大, 由于我们现获取基础资料都是 CAD 格式,在 GIS 里转 shp 后我们会发现很多是地方坐标,对于这种情况,小编也是无能无力,只有想办法 让 CAD 提供方…...

智慧树自动刷课终极解决方案:5分钟告别手动刷课的完整指南

智慧树自动刷课终极解决方案:5分钟告别手动刷课的完整指南 【免费下载链接】zhihuishu 智慧树刷课插件,自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 还在为智慧树平台繁琐的网课学习而烦恼吗&#x…...

RAG分块策略实战:5种方法代码对比+真实业务场景选择指南(附性能测试数据)

RAG分块策略工程实践:5种方法性能对比与场景化选型指南 在构建检索增强生成(RAG)系统时,文档分块策略的选择直接影响着系统的最终效果。本文将深入分析五种主流分块策略的工程实现差异,结合电商客服、医疗问答等典型业…...

麒麟V10系统下微信PC版安装与系统升级全攻略

1. 麒麟V10系统与微信PC版适配现状 最近两年国产操作系统发展迅猛,银河麒麟V10作为其中的佼佼者,已经能够流畅运行微信PC版。但很多用户在安装过程中还是会遇到各种"拦路虎"——找不到安装包、依赖缺失、版本冲突等问题层出不穷。 我实测发现&…...

PX4 EKF滤波效果不好?别只盯着Q和R,这些隐藏参数和传感器预处理同样关键

PX4 EKF滤波效果优化:超越Q/R矩阵的隐藏参数与传感器预处理全解析 当你的无人机在悬停时出现位置漂移,或是穿越机在高速机动时姿态突然发散,大多数开发者第一反应就是调整Q和R矩阵——这就像医生遇到发烧就开退烧药,却忽略了病灶本…...

人工智能工程师应掌握的核心技能与工具

随着人工智能(AI)领域的持续拓展,对专业 AI 工程师的需求呈指数级增长。无论你是刚入行,还是希望实现职业进阶,扎实掌握特定技能与工具都至关重要。本文将详解每位 AI 工程师想要在这一充满活力且竞争激烈的领域立足所…...

OFDRW 2.1.0转换PDF时字体丢失?3种实用解决方案帮你搞定

OFDRW 2.1.0转换PDF字体丢失问题深度解析与实战解决方案 在企业级文档处理系统中,OFD(Open Fixed-layout Document)与PDF之间的格式转换是常见需求。作为国内电子发票、公文交换的标准格式,OFD的准确转换直接关系到业务数据的完整…...

深入剖析Ultralytics中RT-DETR的RepC3模块维度匹配问题

1. RT-DETR与RepC3模块的核心作用 RT-DETR作为Ultralytics推出的实时目标检测模型,其核心优势在于将DETR系列模型的Transformer架构与实时推理需求相结合。我在实际部署中发现,RepC3模块作为模型颈部的关键组件,承担着多尺度特征融合与通道维…...

M5StamPLC工业PLC库:ESP32嵌入式实时控制与I²C外设驱动

1. M5StamPLC 库概述M5StamPLC 是专为 M5Stack 推出的 K141 型号工业级可编程逻辑控制器(PLC)开发板设计的底层驱动库。该板卡并非传统意义上的 Arduino 兼容开发板,而是面向工业自动化场景的嵌入式控制终端,具备数字量输入/输出、…...

ElementUI Table组件实现表头吸顶的进阶技巧与实战

1. 为什么需要表头吸顶功能? 当表格数据量较大时,用户需要滚动页面查看完整内容。这时候如果表头随着滚动消失,用户很容易忘记当前列对应的字段含义,不得不反复回滚查看表头,体验非常糟糕。表头吸顶(Sticky…...

我不是在用 AI 助手,我在把自己的能力沉淀成组织资产淳

1. 什么是 Apache SeaTunnel? Apache SeaTunnel 是一个非常易于使用、高性能、支持实时流式和离线批处理的海量数据集成平台。它的目标是解决常见的数据集成问题,如数据源多样性、同步场景复杂性以及资源消耗高的问题。 核心特性 丰富的数据源支持&#…...

别急着降级!用Anaconda虚拟环境一劳永逸解决Numpy与gensim等库的版本冲突问题

告别版本冲突:用Anaconda虚拟环境彻底解决Python依赖困境 你是否曾在深夜调试代码时,突然遭遇numpy.ndarray size changed这类令人崩溃的二进制兼容性错误?或是花费数小时在不同项目间切换时,反复执行pip uninstall和pip install来…...

Spring IOC 源码学习 事务相关的 BeanDefinition 解析过程 (XML)副

从0构建WAV文件:读懂计算机文件的本质 虽然接触计算机有一段时间了,但是我的视野一直局限于一个较小的范围之内,往往只能看到于算法竞赛相关的内容,计算机各种文件在我看来十分复杂,认为构建他们并能达到目的是一件困难…...

CodeMagicianT源

前面我们对 Kafka 的整体架构和一些关键的概念有了一个基本的认知,本文主要介绍 Kafka 的一些配置参数。掌握这些参数的作用对我们的运维和调优工作还是非常有帮助的。 写在前面 Kafka 作为一个成熟的事件流平台,有非常多的配置参数。详细的参数列表可以…...

新手别怕!用嘉立创EDA两层板搞定ESP8266最小系统PCB(附完整工程文件)

从零开始:用嘉立创EDA轻松打造ESP8266最小系统PCB 第一次打开PCB设计软件时,那种手足无措的感觉我至今记忆犹新。密密麻麻的元件符号、复杂的布线规则、各种专业术语...作为一个刚接触硬件的爱好者,我曾一度怀疑自己是否真的能独立完成一块电…...

ArcGIS密度分析实战:从点、线到核密度的全流程解析

1. 密度分析基础:从概念到应用场景 密度分析是地理信息系统中最常用的空间分析工具之一,它能够将离散的点、线要素转化为连续的密度表面,直观展现空间分布特征。我第一次接触密度分析是在做一个城市商业网点布局项目时,当时需要分…...