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

聊一聊 C# 中的闭包陷阱:foreach 循环的坑你还记得吗?嘲

. 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数字世界厥度豢屯

相关文章:

聊一聊 C# 中的闭包陷阱:foreach 循环的坑你还记得吗?嘲

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

MySQL语句执行深度剖析:从连接到执行的全过程滞

开发个什么Skill呢? 通过 Skill,我们可以将某些能力进行模块化封装,从而实现特定的工作流编排、专家领域知识沉淀以及各类工具的集成。 这里我打算来一次“套娃式”的实践:创建一个用于自动生成 Skill 的 Skill,一是用…...

Ubuntu 配置 Claude Code + MiniMax融

先唠两句:参数就像餐厅点单 把API想象成一家餐厅的“后厨系统”。 ? 路径参数/dishes/{dish_id} -> 好比你要点“宫保鸡丁”这道具体的菜,它是菜单(资源路径)的一部分。查询参数/dishes?spicytrue&typeSichuan -> 好比…...

TMC4671电机驱动调试避坑指南:从SPI通信失败到电机抖动的实战排查

TMC4671电机驱动调试避坑指南:从SPI通信失败到电机抖动的实战排查 调试TMC4671电机驱动芯片时,工程师常会遇到各种"坑"和"雷区"。本文将聚焦实际调试过程中最常见的问题,提供一套从现象到原因的逆向排查方法,…...

可变形卷积实战:从原理到PyTorch实现

1. 可变形卷积的核心原理 第一次接触可变形卷积这个概念时,我正被一个目标检测项目困扰着。传统卷积神经网络在处理形变物体时表现不佳,比如检测不同姿态的行人或者被部分遮挡的车辆。直到发现了可变形卷积这个"黑科技",问题才迎刃…...

Python实战:从零构建天气查询Agent的完整指南

1. 为什么你需要一个天气查询Agent 每次出门前都要手动打开天气App查温度?或者总忘记带伞被突然的暴雨淋成落汤鸡?这些烦恼其实可以用几行Python代码解决。我最近刚用Python给自己写了个天气查询Agent,现在每天早上刷牙时喊一声"查今天天…...

基于MATLAB的GNSS软件接收机跟踪环路详解——自学笔记(3)

1. GNSS软件接收机跟踪环路核心原理 当你第一次打开MATLAB的GNSS软件接收机跟踪函数时,可能会被满屏的变量和运算吓到。别担心,跟踪环路本质上就是个"数字锁匠"——它的任务就是紧紧咬住卫星信号不放。想象一下老式收音机调频,你需…...

基于Newmark法的车桥耦合动力学求解Matlab程序:不平顺车辆-无砟轨道-桥梁耦合全代码研究

车桥耦合matlab程序。 使用newmark法进行数值积分,考虑不平顺车辆-无砟轨道-桥梁耦合的动力学求解全套代码。无砟轨道-桥梁耦合动力学仿真平台—— 基于 Newmark-β 隐式积分的“车-轨-桥”一体化求解框架一、概述无砟轨道桥梁在高速列车通过时表现出强烈的多体-多尺…...

IotNetESP32:面向i-ot.net平台的嵌入式物联网连接抽象库

1. 项目概述IotNetESP32 是一款专为 ESP32 平台设计的嵌入式物联网通信中间件库,其核心定位并非替代底层协议栈,而是构建在 ESP-IDF 或 Arduino-ESP32 框架之上、面向应用层的“连接抽象层”。该库通过封装 WiFi 管理、MQTT 客户端、HTTP 客户端三大基础…...

ADXL362嵌入式驱动库:低功耗加速度计SPI控制与实时采集

1. ADXL362加速度计驱动库技术解析与工程实践ADXL362是Analog Devices(ADI)推出的超低功耗、3轴数字MEMS加速度计,专为电池供电的物联网终端、可穿戴设备、工业状态监测及远程传感器节点等场景设计。其核心优势在于:在全量程2g/4g…...

链栈(链式栈) 超详细实现(C 语言 + 逐行精讲)

前言栈(Stack) 是一种后进先出(LIFO)的线性数据结构。前面我们学习了顺序栈(数组实现),今天我们学习它的兄弟 ——链栈(链式栈)。链栈 用单链表实现的栈它完美解决了顺序…...

代码审查指南高效协作与质量保证

代码审查指南:高效协作与质量保证 在软件开发过程中,代码审查是确保代码质量、提升团队协作效率的关键环节。通过系统化的审查流程,团队能够及早发现潜在缺陷,统一代码风格,并促进知识共享。缺乏规范的审查流程可能导…...

Akafugu TWILiquidCrystal:I²C LCD驱动库详解与工程实践

1. Akafugu TWILiquidCrystal 库概述Akafugu TWILiquidCrystal 是一套专为 Akafugu 公司设计的 TWI/IC 接口 LCD 控制器开发的固件与 Arduino 软件库组合方案。该方案的核心目标是以极简硬件连接(仅需 4 根线)和低资源开销,实现对标准 HD4478…...

LoRaWAN网络部署实战指南:从规划到优化的全链路解析

1. LoRaWAN网络部署前的关键思考 第一次接触LoRaWAN网络部署时,很多人会直接跳到网关选型环节,这其实是个常见误区。我在实际项目中见过不少团队因为前期规划不足,导致后期频繁调整网络架构,既浪费资源又影响项目进度。部署LoRaWA…...

Jenkins 学习总结滩

先唠两句:参数就像餐厅点单 把API想象成一家餐厅的“后厨系统”。 ? 路径参数/dishes/{dish_id} -> 好比你要点“宫保鸡丁”这道具体的菜,它是菜单(资源路径)的一部分。 查询参数/dishes?spicytrue&typeSichuan -> …...

macos简单配置openclaw诼

1 实用案例 1.1 表格样式生成 本示例用于生成包含富文本样式与单元格背景色的Word表格文档。 模板内容: 渲染代码: # python-docx-template/blob/master/tests/comments.py from docxtpl import DocxTemplate, RichText # data: python-docx-template/bl…...

Python AI爬虫实战:爬取张雪峰微博并进行情感分析与词云可视化宜

1. 引入 在现代 AI 工程中,Hugging Face 的 tokenizers 库已成为分词器的事实标准。不过 Hugging Face 的 tokenizers 是用 Rust 来实现的,官方只提供了 python 和 node 的绑定实现。要实现与 Hugging Face tokenizers 相同的行为,最好的办法…...

龙芯k - 走马观碑组ST驱动移植柏

正文 异步/等待解决了什么问题? 在传统同步I/O操作中(如文件读取或Web API调用),调用线程会被阻塞直到操作完成。这在UI应用中会导致界面冻结,在服务器应用中则造成线程资源的浪费。async/await通过非阻塞的异步操作解…...

RegisterWriter:ROHM传感器寄存器操作的C++类型安全封装库

1. RegisterWriter 库概述RegisterWriter 是 ROHM 半导体为加速其传感器驱动开发而设计的一套轻量级 C 硬件抽象层(HAL)辅助库,核心定位并非替代标准 MCU HAL(如 STM32 HAL 或 Nordic nRF SDK),而是作为寄存…...

从技术到管理:一名一线开发者的转型心路历程

从技术到管理:一名一线开发者的转型心路历程 在技术行业,许多开发者都曾面临一个关键选择:是继续深耕技术,还是转型为管理者?这条转型之路充满挑战与成长,既是对个人能力的考验,也是对职业规划…...

AI Agent Harness Engineering 的商业化困局:按 Token 计费与按结果付费的博弈

从零破解AI Agent Harness商业化生死门:Token计费惯性与结果付费终局的双向奔赴与博弈深度 副标题:从代码层面解构Agent开发成本模型,从商业落地剖析价值定价逻辑,构建兼顾技术可行性、客户信任度与ROI的可持续盈利体系 第一部分:引言与基础 (Introduction & Foundati…...

多品类迷雾:为何亚马逊店铺无法用“宽泛口号”建立有效定位

当一个品牌或店铺像福特汽车一样,横跨多个品类和型号时,便面临一个根本性的定位困境:它无法在任何一个具体的品类中建立“专家”认知,因此被迫退回到寻找一个覆盖所有产品的“最大公约数”——通常是一个宽泛、无力、难以验证的抽…...

品牌基因烙印:在亚马逊,为何成功的旧名字会成为转型的最大障碍

在商业世界中,一个公司的名字是其最核心的“心智基因”,一旦形成便极难改变。正如“普尔曼”永远让人想起火车车厢,“灰狗”即是长途客运的代名词,即使它们的业务早已多元,巨额的广告也无法扭转公众的固化认知。在亚马逊,这一规律被算法和搜索行为进一步放大:一个在特定…...

无形估值:在亚马逊,为何“公司定位”是你吸引顶级资源的核心资产

“公司的买卖”不仅发生在并购交易中,更持续发生在每一次关键资源向你靠拢的瞬间。在亚马逊的生态中,这表现为:当顶尖人才考虑加入、优质工厂寻求合作、行业资本决定投资、或平台给予流量扶持时,他们本质上都在“购买”你公司未来…...

STM32F1轻量级USB HID键盘鼠标复合设备固件库

1. 项目概述KeyboardMouse 是一个面向 STM32F1 系列微控制器的轻量级 USB HID(Human Interface Device)固件库,专为实现复合型 USB 键盘与鼠标设备而设计。该库不依赖第三方 USB 协议栈(如 ST 的 USB Device Library 或 Keil ARM …...

BMP183气压传感器驱动开发与高精度补偿实践

1. BMP183气压传感器驱动库技术解析与工程实践BMP183是由博世(Bosch)推出的高精度数字气压传感器,广泛应用于无人机高度计、气象站、可穿戴设备及工业环境监测等嵌入式系统中。该器件集成MEMS压阻式压力传感单元、温度传感元件及24位ADC&…...

《空间智能体:下一代AI基础设施》——从视觉识别到空间计算的范式跃迁

《空间智能体:下一代AI基础设施》——从视觉识别到空间计算的范式跃迁摘要(Abstract)近年来,人工智能系统在视觉识别、目标检测与多目标跟踪等任务中取得显著进展。然而,大量研究与工程实践表明,传统基于图…...

KY040旋转编码器驱动详解:消抖、正交解码与多平台适配

1. KY040-rotary 库深度解析:面向嵌入式工程师的旋转编码器驱动实践指南旋转编码器是人机交互中最基础、最可靠的物理输入设备之一,广泛应用于工业控制面板、音频设备音量调节、仪器仪表参数设置等场景。KY-040(亦称 HW-040)作为一…...

SparkFun AVR ISP编程库:嵌入式量产级AVR烧录实现

1. SparkFun AVR ISP 编程库深度解析:面向嵌入式量产的底层ISP烧录实现1.1 库定位与工程价值SparkFun AVR ISP Programming Library 是一个轻量级、零依赖的纯C底层编程库,专为在嵌入式主控(如Arduino兼容板)上实现对AVR微控制器&…...

AVR-IoT Cellular Mini底层技术解析:安全蜂窝连接与低功耗设计

1. AVR-IoT Cellular Mini 开发板底层技术解析AVR-IoT Cellular Mini 是 Microchip 推出的面向蜂窝物联网(Cellular IoT)应用的紧凑型开发平台,其核心价值不仅在于硬件集成度,更在于其构建在 DxCore 基础上的完整 Arduino 兼容软件…...