当前位置: 首页 > 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) 组成的&…...

TP-Link 多款路由器曝未修复零日漏洞:栈溢出可致远程代码执行,其他漏洞已被实际利用

目前,TP-Link 已确认多款路由器型号存在尚未修复的零日漏洞,同时该品牌其他漏洞已被真实网络攻击利用。 Amazon.com: TP-Link Archer AX10 AX1500 WiFi 6 Router Dual Band 1.5GHz Tri Core CPU TPLink : Electronics 零日漏洞详情与厂商响应 该零日漏…...

WarcraftHelper:三步解决魔兽争霸III在现代电脑上的兼容性问题

WarcraftHelper:三步解决魔兽争霸III在现代电脑上的兼容性问题 【免费下载链接】WarcraftHelper Warcraft III Helper , support 1.20e, 1.24e, 1.26a, 1.27a, 1.27b 项目地址: https://gitcode.com/gh_mirrors/wa/WarcraftHelper 还在为经典魔兽争霸III在现…...

RI-CLPM模型中的协变量控制:显变量水平 vs 随机截距水平(Mplus语法详解)

RI-CLPM模型中协变量控制的策略选择:显变量水平与随机截距水平的深度对比 在纵向数据分析领域,随机截距交叉滞后模型(RI-CLPM)因其能够区分个体间差异和个体内变化而广受欢迎。然而,当研究者需要在模型中纳入协变量时,往往会面临…...

Hexo Admin实战指南:打造高效本地Markdown博客管理后台

1. 为什么你需要Hexo Admin? 如果你正在使用Hexo搭建静态博客,肯定遇到过这样的烦恼:每次写新文章都要打开终端,输入hexo new post "文章标题",然后在生成的Markdown文件里手动编辑。这种操作不仅繁琐&…...

看Anything V5如何玩转AI绘画:从简单描述到复杂场景的生成效果案例

看Anything V5如何玩转AI绘画:从简单描述到复杂场景的生成效果案例 如果你对AI绘画感兴趣,一定听说过Stable Diffusion。而Anything V5,就是基于这个强大框架的一个特别版本,它在动漫、二次元风格的图像生成上表现尤为出色。今天…...

YOLO11实例分割教程:快速掌握数据标注、格式转换与模型训练

YOLO11实例分割教程:快速掌握数据标注、格式转换与模型训练 1. 准备工作与环境搭建 1.1 获取YOLO11镜像 YOLO11镜像提供了完整的计算机视觉开发环境,包含预装好的所有依赖项。您可以通过以下两种方式使用: Jupyter Notebook方式&#xff1…...

MT4跟单系统高频交易优化:如何用Pumping模式降低服务器负载50%

MT4跟单系统高频交易优化:Pumping模式实战解析与性能提升方案 外汇交易市场瞬息万变,对于专业交易团队而言,毫秒级的延迟可能意味着巨大的利润差异。在MT4跟单系统中,传统轮询方式在高频交易场景下往往成为性能瓶颈,导…...

从LED闪烁到继电器驱动:手把手用Arduino玩转NPN/PNP三极管开关电路(附代码)

从LED闪烁到继电器驱动:手把手用Arduino玩转NPN/PNP三极管开关电路(附代码) 在创客项目和物联网设备开发中,我们常常会遇到单片机IO口驱动能力不足的问题。比如当你想要控制一个高功率LED、蜂鸣器或者继电器时,Arduino…...

深度学习图像拼接新突破:USID++如何实现无监督大视差场景下的精准对齐

1. 为什么传统图像拼接技术会翻车? 想象一下你正在用手机拍摄一张全景照片,从左往右缓慢移动镜头。当你把两张照片拼在一起时,近处的树木和远处的山峦经常会出现"鬼影"或错位——这就是典型的视差问题。传统图像拼接方法在这个场景…...

双目视觉测量系统在工业检测中的精度优化策略与实践

1. 双目视觉测量系统在工业检测中的核心价值 在工业质检领域,毫米级的精度差异可能直接决定产品合格率。去年我们团队为某汽车零部件厂部署检测系统时,就遇到过螺栓螺纹检测误判的难题——传统单目相机总是把0.2mm的螺纹瑕疵漏检。换成双目系统后&#x…...

Win11 Docker Desktop 迁移虚拟硬盘文件存储位置

一、wsl虚拟硬盘文件路径 C:\Users\admin\AppData\Local\Docker\wsl C:\Users\admin\AppData\Local\Docker\wsl\disk\docker_data.vhdx C:\Users\admin\AppData\Local\Docker\wsl\main\ext4.vhdx 二、新建新的磁盘映像位置 E:\wsl2(选择这个) E:\ws…...

Sollumz:3步在Blender中制作GTA V游戏模组的完整指南

Sollumz:3步在Blender中制作GTA V游戏模组的完整指南 【免费下载链接】Sollumz Grand Theft Auto V modding suite for Blender. This add-on allows the creation of modded game assets: 3D models, maps, interiors, animations, etc. 项目地址: https://gitco…...

商务本也能跑AI!手把手教你用Ollama+Chatbox在ThinkPad上免费部署DeepSeek-R1

商务本也能跑AI!手把手教你用OllamaChatbox在ThinkPad上免费部署DeepSeek-R1 当大多数人还在为运行AI模型需要高端显卡发愁时,你可能不知道,手边的商务笔记本就能开启本地AI之旅。作为一名常年与ThinkPad X1 Carbon为伴的技术顾问&#xff0c…...

SD卡接口PCB设计实战:从引脚定义到高速信号完整性布局布线

1. SD卡接口基础:从物理结构到引脚定义 第一次接触SD卡接口设计时,我对着那排密密麻麻的引脚直发懵。后来才发现,理解SD卡物理结构是PCB设计的第一步。常见的SD卡有标准SD、microSD(TF卡)和miniSD三种规格,…...

抖音无水印下载器终极指南:三步快速获取高清内容的完整教程

抖音无水印下载器终极指南:三步快速获取高清内容的完整教程 【免费下载链接】douyin-downloader A practical Douyin downloader for both single-item and profile batch downloads, with progress display, retries, SQLite deduplication, and browser fallback …...

逆向思维看保护:我是如何用VMProtect SDK给自己的工具软件“上锁”,并防止被破解的?

逆向思维构建软件护城河:VMProtect SDK实战防御手册 当我在深夜调试自己开发的工具软件时,突然冒出一个念头:如果我是黑客,会如何破解这个软件?这个看似简单的自问,彻底改变了我对软件保护的认知方式。传统…...

【华为电脑管家】多屏协同下微软拼音输入法兼容性自动修复的终极指南

1. 多屏协同与输入法兼容性问题解析 华为电脑管家的多屏协同功能确实让跨设备办公变得无比便捷,但很多用户都遇到过这个烦人的问题:每次连接多屏协同时,微软拼音输入法的兼容性设置就会被自动修改。我自己也深受其扰,经常在重要会…...

BAAI/bge-m3实战:快速构建个人知识库与智能问答助手

BAAI/bge-m3实战:快速构建个人知识库与智能问答助手 1. 项目概述与核心价值 BAAI/bge-m3是北京智源研究院推出的开源语义嵌入模型,在MTEB(Massive Text Embedding Benchmark)榜单上表现优异。这个多语言通用嵌入模型能够将文本转…...

全球AI监管格局:合规将成为企业AI落地的核心门槛

全球AI监管现状欧盟《人工智能法案》将AI系统分为四类风险等级,禁止不可接受风险类AI(如社会评分系统),高风险类AI需满足严格合规要求(如医疗设备)。违规罚款可达全球营业额6%。美国采取分行业监管模式&…...

AI算力行业深度报告:供需格局、技术演进与投资机会

AI算力行业概述AI算力指支撑人工智能模型训练和推理所需的计算资源,核心包括芯片、服务器、数据中心等硬件设施。随着大模型技术爆发,全球算力需求呈现指数级增长,预计2030年市场规模将突破万亿美元。供需格局分析供给端芯片领域:…...

从通用到垂直:行业大模型将成为企业数字化转型的核心抓手

行业大模型的崛起背景数字化转型进入深水区,企业对AI的需求从通用场景转向垂直领域。通用大模型在特定行业中面临数据敏感性、专业知识不足、成本过高等问题,催生了行业大模型的快速发展。行业大模型的差异化优势精准性:针对行业数据训练&…...

低空经济“充电网”:原理、场景与未来布局全解析

低空经济“充电网”:原理、场景与未来布局全解析 引言:为什么说“充电桩”是低空经济的“加油站”? [外链图片转存中…(img-5rpT3Icb-1775923220357)] 随着无人机与eVTOL(电动垂直起降飞行器)从“玩具”和“概念”走向…...

Sollumz:在Blender中打造专业级GTA V游戏资产的终极指南 [特殊字符]

Sollumz:在Blender中打造专业级GTA V游戏资产的终极指南 🎮 【免费下载链接】Sollumz Grand Theft Auto V modding suite for Blender. This add-on allows the creation of modded game assets: 3D models, maps, interiors, animations, etc. 项目地…...

Input Leap架构深度解析:跨平台KVM软件的技术实现与多设备输入协同

Input Leap架构深度解析:跨平台KVM软件的技术实现与多设备输入协同 【免费下载链接】input-leap Open-source KVM software 项目地址: https://gitcode.com/gh_mirrors/in/input-leap 在现代多设备办公环境中,Input Leap作为一款开源的KVM&#x…...

高效实现分组内跨行时间戳匹配:构建 user_rejects 布尔标识列

本文介绍如何在大规模数据集(百万级行、每组15–25行)中,基于 application_id 分组,高效判断每行的 rejected_time 是否等于同组内任意其他行的 selected_time,并生成整数型布尔列 user_rejects。 本文介绍如何在…...

如何中止正在运行的RMAN备份_利用OS kill进程或SQL强杀通道会话

中止RMAN备份必须使用RMAN自身命令(如ABORT)或CtrlC,禁用kill -9;否则易致控制文件损坏,引发ORA-00205或ORA-00600错误,后续须验证v$backup_set、执行CROSSCHECK及备份控制文件。中止 RMAN 备份时&#xff…...

BepInEx插件框架实战指南:构建高效稳定的Unity游戏模组生态系统

BepInEx插件框架实战指南:构建高效稳定的Unity游戏模组生态系统 【免费下载链接】BepInEx Unity / XNA game patcher and plugin framework 项目地址: https://gitcode.com/GitHub_Trending/be/BepInEx BepInEx作为Unity Mono、IL2CPP和.NET框架游戏的强大插…...

考虑需求响应和碳交易的柔性负荷综合能源系统优化调度模型

考虑需求响应和碳交易的综合能源系统日前优化调度模型 关键词:柔性负荷 需求响应 综合能源系统 参考:私我 仿真平台:MATLAB yalmipcplex 主要内容:在冷热电综合能源系统的基础上,创新性的对用户侧资源进行了细致的划…...

基于Matlab和Cplex的微电网优化调度研究:涵盖风光热储能及多场景负荷模拟分析

考虑风光火储的微电网优化调度 软件:Matlabcplex 介绍:考虑风电、光伏、热电机组和储能优化调度,其中负荷考虑冬季或夏季两种场景,并且考虑晴天、多云、雨天、多风和少风场景,对风机考虑相应的故障概率,以火…...