源码分享-golang的BMP文件读写库

用于读写BMP文件的golang版源码

源码基于源码分享-golang的二进制文件读写库 https://blog.csdn.net/zhyulo/article/details/128890546

BMP文件格式可参考位图文件解析-位图(bmp)、图标(ico)与光标(cur) https://blog.csdn.net/zhyulo/article/details/85934728

import ("binary""bufio""errors""fmt""os"
)var ofb = errors.New("out of bound")type BitmapFile struct {Header   BitmapFileHeaderInfo     BitmapInfoHeaderQuad     BitmapQuad  `info:"调色板"`BitLines BitmapLines `info:"点阵数据"`
}func ReadBitmapFile(path string) (*BitmapFile, error) {buf, err := os.ReadFile(path)if err != nil {return nil, err}var bmp BitmapFileerr = binary.Unmarshal(buf, false, &bmp)return &bmp, err
}func NewBitmapFile(width, height int32, bitCount uint16) *BitmapFile {bmp := &BitmapFile{Header: BitmapFileHeader{Type: MagicBmp{'B', 'M'}},Info: BitmapInfoHeader{Size:     40,Width:    width,Height:   height,Planes:   1,BitCount: bitCount,},}if height < 0 {height = -height}bmp.BitLines = make(BitmapLines, height, height)bytePerLine := (int32(bitCount)*width + 31) / 32 * 4for i := int32(0); i < height; i++ {bmp.BitLines[i] = make(BitmapLine, bytePerLine, bytePerLine)}bmp.Info.SizeImage = uint32(bytePerLine * height)if bitCount < 16 {bmp.Info.ClrUsed = 1 << bitCountbmp.Quad = make(BitmapQuad, bmp.Info.ClrUsed, bmp.Info.ClrUsed)}bmp.Header.OffBits = 14 + bmp.Info.Size + uint32(len(bmp.Quad))*4bmp.Header.Size = bmp.Header.OffBits + bmp.Info.SizeImagereturn bmp
}func (v *BitmapFile) SetQuad(f func(index int) RgbQuad) {for i := 0; i < len(v.Quad); i++ {v.Quad[i] = f(i)}
}func (v *BitmapFile) ReadLine(y int) ([]int, error) {height := v.Info.Heightif height < 0 {height = -height}if y < 0 || y >= int(height) {return nil, ofb}val := make([]int, v.Info.Width, v.Info.Height)err := binary.NewDecoder(v.BitLines[y], 0).Decode(val, true, int(v.Info.BitCount))return val, err
}func (v *BitmapFile) WriteLine(y int, val any) error {height := v.Info.Heightif height < 0 {height = -height}if y < 0 || y >= int(height) {return ofb}w := &lineWriter{buf: v.BitLines[y]}enc := binary.NewEncoder(w)defer enc.Write(nil)return enc.Encode(val, true, int(v.Info.BitCount))
}func (v *BitmapFile) Save(path string) error {f, err := os.Create(path)if err != nil {return err}defer f.Close()buf := bufio.NewWriter(f)defer buf.Flush()enc := binary.NewEncoder(buf)defer enc.Write(nil)return enc.Encode(v, false, 0)
}type BitmapFileHeader struct {Type      MagicBmp `info:"文件标识"`Size      uint32   `info:"文件大小"`Reserved1 uint16Reserved2 uint16OffBits   uint32 `info:"点阵数据偏移"`
}type MagicBmp [2]bytefunc (v *MagicBmp) UnmarshalBinary(dec *binary.Decoder, isBig bool, bit int) error {err := dec.Decode(v[:], isBig, bit)if err != nil {return err}if string(v[:]) != "BM" {return errors.New("file isn't bitmap format")}return nil
}type BitmapInfoHeader BitmapInfo
type BitmapInfo struct {Size          uint32    `info:"信息头大小"`Width         int32     `info:"宽度(像素)"`Height        int32     `info:"高度(像素)"`Planes        uint16    `info:"颜色平面数"`BitCount      uint16    `info:"比特/像素"`Compression   uint32    `info:"压缩类型"`SizeImage     uint32    `info:"点阵数据大小"`XPelsPerMeter int32     `info:"水平分辨率(像素/米)"`YPelsPerMeter int32     `info:"垂直分辨率(像素/米)"`ClrUsed       uint32    `info:"调色板颜色数"`ClrImportant  uint32    `info:"关键颜色数"`ClrMask       ColorMask `info:"颜色掩码"`
}func (v *BitmapInfoHeader) UnmarshalBinary(dec *binary.Decoder, isBig bool, bit int) error {dec.SetArg(v)return dec.Decode((*BitmapInfo)(v), isBig, bit)
}func (v *BitmapInfoHeader) MarshalBinary(enc *binary.Encoder, isBig bool, bit int) error {enc.SetArg(v)return enc.Encode((*BitmapInfo)(v), isBig, bit)
}type ColorMask [3]uint32func (v *ColorMask) UnmarshalBinary(dec *binary.Decoder, isBig bool, bit int) error {bmp, ok := dec.Arg().(*BitmapInfoHeader)if !ok || bmp == nil {return errors.New("decode arg not set")}if bmp.Size == 40 {return nil} else if bmp.Size != 52 {return fmt.Errorf("unsupport BitmapInfoHeader size: %d", bmp.Size)}return dec.Decode(v[:], isBig, bit)
}func (v *ColorMask) MarshalBinary(enc *binary.Encoder, isBig bool, bit int) error {bmp, ok := enc.Arg().(*BitmapInfoHeader)if !ok || bmp == nil {return errors.New("encode arg not set")}if bmp.Size == 40 {return nil} else if bmp.Size != 52 {return fmt.Errorf("unsupport BitmapInfoHeader size: %d", bmp.Size)}return enc.Encode(v[:], isBig, bit)
}type BitmapQuad []RgbQuad
type RgbQuad struct {Blue     byteGreen    byteRed      byteReserved byte
}func (v *BitmapQuad) UnmarshalBinary(dec *binary.Decoder, isBig bool, bit int) error {bmp, ok := dec.Arg().(*BitmapInfoHeader)if !ok || bmp == nil {return errors.New("decode arg not set")}if bmp.BitCount >= 16 {return nil}num := bmp.ClrUsedif num == 0 {num = 1 << bmp.BitCount}*v = make(BitmapQuad, num, num)return dec.Decode((*v)[:], isBig, bit)
}type BitmapLines []BitmapLine
type BitmapLine []bytefunc (v *BitmapLines) UnmarshalBinary(dec *binary.Decoder, isBig bool, bit int) error {bmp, ok := dec.Arg().(*BitmapInfoHeader)if !ok || bmp == nil {return errors.New("decode arg not set")}bytePerLine := int((int32(bmp.BitCount)*bmp.Width + 31) / 32 * 4)lineNum := int(bmp.Height)if lineNum < 0 {lineNum = -lineNum}*v = make([]BitmapLine, lineNum)for i := 0; i < lineNum; i++ {newDec := dec.SubDecoder(bytePerLine)err := newDec.Decode(&(*v)[i], isBig, bit)if err != nil {return err}err = dec.Seek(newDec.Pos())if err != nil {return err}}return nil
}type lineWriter struct {buf []byteoft int
}func (w *lineWriter) Write(p []byte) (n int, err error) {n = len(p)if w.oft+n > len(w.buf) {n = len(w.buf) - w.ofterr = ofb}copy(w.buf[w.oft:], p)w.oft += nreturn
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/638538.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

一文梳理金融风控建模全流程(Python)

▍目录 一、简介 风控信用评分卡简介 Scorecardpy库简介 二、目标定义与数据准备 目标定义 数据准备 三、安装scorecardpy包 四、数据检查 五、数据筛选 六、数据划分 七、变量分箱 卡方分箱 手动调整分箱 八、建立模型 相关性分析 多重共线性检验VIF KS和AUC …

李宏毅 Generative Adversarial Network(GAN)生成对抗网络

(延申)GAN Lecture 1 (2018)- Introduction_哔哩哔哩_bilibili Basic Idea of GAN 附课程提到的各式各样的GAN&#xff1a;https://github.com/hindupuravinash/the-gan-zoo 想要让机器做到的是生成东西。->训练出来一个generator。 假设要做图像生成&#xff0c;要做的是…

【Qt之模型视图】5. Qt库提供的视图便捷类

1. 前言 Qt提供了一些便捷的视图类&#xff0c;如QListWidget提供项目列表&#xff0c;QTreeWidget显示多级树结构&#xff0c;QTableWidget提供单元格项的表格。每个类都继承了QAbstractItemView类的行为&#xff0c;QAbstractItemView类实现了项目选择和标题管理的通用行为。…

16.5 参考文献——深度学习定位

16.5 一种高效鲁棒的多楼层室内环境指纹定位方法 同济大学 Zhao Y, Gong W, Li L, et al. An Efficient and Robust Fingerprint Based Localization Method for Multi Floor Indoor Environment[J]. IEEEa Internet of Things Journal, 2023. 2.相关工作 B.基于深度学习的…

固态硬盘优化设置

目录 前言&#xff1a; 关闭Windows Search 禁用系统保护&#xff08;不建议&#xff09; 不建议禁用系统保护原因 关闭碎片整理【机械硬盘】 提升固态硬盘速度 开启TRIM 合理使用固态硬盘的容量 正确关机 关闭开机自启 前言&#xff1a; 电脑配备固态硬盘就能一劳…

自我摸索:如何运营并玩转CSDN?

自注册CSDN以来已有七年之久&#xff0c;但真正运营CSDN也是最近一年的事情&#xff0c;大概就是22年底&#xff0c;参加2022 博客之星 的竞选。接触了很多大佬&#xff0c;也学习模仿着开始玩转CSDN&#xff0c;虽然没有同期运营的大佬们玩的6&#xff0c;但也有一些经验可以来…

Arduino 智能学习简易版 机械臂

文章目录 一 总体思路1 功能原理2 硬件准备3 Arduino库 二 代码设计1 舵机部分2 电位器3 记录路径法4 触控板与三色灯板 三 展示效果视频链接 一 总体思路 1 功能原理 实现功能&#xff1a;使用电位器控制每一个舵机运动&#xff0c;进而操作机械臂&#xff1b;设置学习模式&…

this指针相关

一、类对象的存储方式 先说一下结论&#xff1a;一个类的大小&#xff0c;实际就是该类中”成员变量”之和&#xff0c;当然要注意内存对齐 注意空类和没有成员变量的类的大小&#xff0c;编译器给了这两种类一个字节来唯一标识这个类的对象。 class A { public:int _a;int _b…

【RL】(task3)A2C、A3C算法、JoyRL

note 文章目录 note一、A2C算法二、A3C算法时间安排Reference 一、A2C算法 在强化学习中&#xff0c;有一个智能体&#xff08;agent&#xff09;在某种环境中采取行动&#xff0c;根据其行动的效果获得奖励。目标是学习一种策略&#xff0c;即在给定的环境状态下选择最佳行动&…

Python多线程—threading模块

参考&#xff1a;《Python核心编程》 threading 模块的Thread 类是主要的执行对象&#xff0c;而且&#xff0c;使用Thread类可以有很多方法来创建线程&#xff0c;这里介绍以下两种方法&#xff1a; 创建 Thread 实例&#xff0c;传给它一个函数。派生 Thread 的子类&#xf…

计算机体系结构——多处理机系统

一、概述 重要概念 评估指标 通信延迟 通信延迟&#xff1d;发送开销&#xff0b;跨越时间&#xff0b;传输延迟&#xff0b;接收开销 跨越时间 数字信号从发送方的线路端传送到接收方的线路端所经过的时间。 传输时间 全部的消息量除以线路带宽。 多处理机的架构 根…

Unity 之 Addressable可寻址系统 -- HybridCLR(华佗)+Addressable实现资源脚本全热更 -- 实战(二)

Unity 之 Addressable可寻址系统 -- HybridCLRAddressable实现资源脚本全热更 -- 实战 前言实现原理一&#xff0c;HybridCLR相关操作1.1 安装HybridCLR1.2 打包程序集1.2 设置面板1.3 补充元数据1.4 代码加载流程 二&#xff0c;Addressable资源管理2.1 生成热更代码资源2.2 创…

【0246】深入分析PG内核Write-Ahead Log的实现机制

文章目录 1. 前言2. Write-Ahead Log2.1 WAL数据结构2.1.1 逻辑结构(Logical Structure)2.1.2 物理结构(Physical Structure)3. 参考文献1. 前言 Write-Ahead Log简写为WAL,即预写式日志,PG数据库内核引入WAL的原因基于以下几点: (1)如果发生停电、操作系统错误或数据…

攻防世界——answer_to_everything-writeup

__int64 __fastcall not_the_flag(int a1) {if ( a1 42 )puts("Cipher from Bill \nSubmit without any tags\n#kdudpeh");elseputs("YOUSUCK");return 0LL; } kdudpeh这个东西&#xff0c;根据题目提示sha1加密 import hashlib flagkdudpeh x hashlib…

网安防御保护入门

常见的网络安全术语&#xff1a; 漏洞&#xff08;脆弱性&#xff09;&#xff1a;可能被一个或多个威胁利用的资产或控制的弱点 攻击&#xff1a;企图破坏、泄露、篡改、损伤、窃取、未授权访问或未授权使用资产的行为 入侵&#xff1a;对网络或联网系统的未授权访问&#xff…

【Springboot】日志

1.日志的使用 日志主要用于记录程序运行的情况。我们从学习javase的时候就使用System.out.println();打印日志了&#xff0c;通过打印的日志来发现和定位问题&#xff0c;或根据日志来分析程序运行的过程。在Spring的学习中,也经常根据控制台的⽇志来分析和定位问题 。 日志除…

k8s-pvc/pv扩容记录

背景 一次聊天过程中&#xff0c;对方提及pvc的扩容&#xff0c;虽然有注意过 storageclass 有个AllowVolumeExpansion的配置&#xff08;有些csi插件是不支持该配置的&#xff0c;比如local-volume-provisoner&#xff09;&#xff0c;但是没有实际用过&#xff0c;所以还是心…

数据分析实战:城市房价分析

流程图&#xff1a; 1.读数据表 首先&#xff0c;读取数据集。 CRIMZNINDUSCHASNOXRMAGEDISRADTAXPTRATIOBLSTATtarget0.00632182.3100.5386.57565.24.09129615.3396.94.98240.0273107.0700.4696.42178.94.9671224217.8396.99.1421.60.0272907.0700.4697.18561.14.9671224217…

【MySQL数据库专项 一】一个例子讲清楚数据库三范式

好的&#xff0c;让我们以学校数据库中的一个表为例来说明第一范式&#xff08;1NF&#xff09;、第二范式&#xff08;2NF&#xff09;和第三范式&#xff08;3NF&#xff09;的概念。 什么是数据库三范式 数据库的范式&#xff08;Normalization&#xff09;是一组关于数据…

UI -- Vue2

Vue2 模板 Vue2 UI package.json同级components/slButton/element.vue <template><div class"d1"><span>测试123</span></div> </template><script>export default {name:"sl-button",data() {return {};}} …