利用C/C++二进制读写png文件的方法示例

 更新时间:2020年4月25日 17:33  点击:2059

前言

二进制文件不是以ASCII代码存放数据的,它将内存中数据存储形式不加转换地传送到磁盘文件,因此它又称为内存数据的映像文件。因为文件中的信息不是字符数据,而是字节中的二进制形式的信息,因此它又称为字节文件。

对二进制文件的操作也需要先打开文件,用完后要关闭文件。在打开时要用ios::binary指定为以二进制形式传送和存储。二进制文件除了可以作为输入文件或输出文件外,还可以是既能输入又能输出的文件。这是和ASCII文件不同的地方。

需求

最近为了弄OpenGl的纹理代码,发现书上没有图片像素的获取,然后就想写个来获取png的,结果花了一天的时间没弄清楚为什么出现数据个别正确其他的却是205

突然想起来以前弄软工的时候虽然那个网站只完成了登入注册和文本显示,但是想在数据库中存储图片的时候了解到1存图片地址,2存图片二进制数据。

没错就是二进制。然后拿起C++的翻开找啊找,弄了个ifstream iOS::binary的,成功数据正常。
时隔一天才又想起来r和rb好像是有区别的。没错,那些知识确实没有记住。然后就把C的也改ok了

以下代码只有最简单的读写。地址定位啥的,个别注释中有。如果要改动png的格式甚么的就要再了解一下png的数据结构
如果要十进制的话就跟着注释改一下

mm.png

实例代码如下

#include<iostream> 
#include<fstream> 
using namespace std; 
typedef unsigned char byte; 
 /* 
class PngMsg 
{ 
private : 
 unsigned char markMsg[8]; //十进制,相当于16进制89.50.4e.47.0d.0a.1a.0a; 
 char widthloc; 
 char heigtMsgloc; 
 char BitDepthloc;//图像深度 
 char ColorTypeloc; 
 char CompressionMethodloc;//压缩方法(LZ77派生算法) 
 char FilterMethodloc;//滤波器方法 
 char InterlaceMethodloc; 
public: 
 PngMsg() 
 { 
 markMsg[0] = 137;markMsg[1] = 80; markMsg[2] = 78;markMsg[3] = 71; markMsg[4] = 13;markMsg[5] = 10; markMsg[6] = 26; markMsg[7] = 10; 
 widthloc = 'a'; 
 heigtMsgloc = 'b'; 
 BitDepthloc = 'c';//图像深度 
 ColorTypeloc = 'd'; 
 CompressionMethodloc = 'e';//压缩方法(LZ77派生算法) 
 FilterMethodloc = 'f';//滤波器方法 
 InterlaceMethodloc = 'g'; 
 } 
 long int getMsg(char loc) 
 { 
 if (loc == 'a')return 0x10; 
 if (loc == 'b')return 0x14; 
 if (loc == 'c')return 0x15; 
 if (loc == 'd')return 0x16; 
 if (loc == 'e')return 0x17; 
 if (loc == 'f')return 0x18; 
 if (loc == 'g')return 0x19; 
 } 
 unsigned char width[4];//图像宽度,单位像素 
 unsigned char height[4];//图像高度,单位像素 
 unsigned char BitDepth; 
 //图像深度 
 //索引彩色1.2.4.8;灰度1.2.4.8.16;真彩色8.16 
 unsigned char ColorType; 
 //0灰度1.2.4.8.16;2真彩色8.16;3索引彩色1.2.4.8 
 //4带α通道数据的灰度8.16;6带α通道数据的真彩色8.16 
 unsigned char CompressionMethod;//压缩方法(LZ77派生算法) 
 unsigned char FilterMethod;//滤波器方法 
 unsigned char InterlaceMethod;//0:非隔行扫描;1:Adam7 
};*/ 
 
 
//=============================== 
 
 
//=============== 
//二进制读入。书上写ASCII码读取和二进制读取,如果对象是字母,那么一致。如果是数字,那么不一致 
//书中说明【文件中数据的组织形式,分为ASCII文件(一个字节存放一个ASCII代码)和二进制文件(内部文件,存储形式原样在磁盘上存放),】 
//字符,内存存储=ASCII=二进制形式 
//数值数据,内存存储和ASCII码不同。 
//样例内存整数100000. 
//---------------------------------------------------------------- 
//内存地址 0x00 01 02 03 
//内存 00000000 00000000 00100111 00010000【大端模式下】 
//---------------------------------------------------------------- 
//二进制 00000000 00000000 00100111 00010000 
//---------------------------------------------------------------- 
//ASCII 00110001 00110000 00110000 00110000 00110000 00110000【6个字节】 
//ASCII码对应 1的49 0的48 0的48 0的48 0的48 0的48 
//---------------------------------------------------------------- 
 
 
 
//只有含‘写'的不存在的文件会新建,其他会报错 
 
//r只读;w只写;a尾增(附加/写);文本ASCII 
//rb读;wb写;ab尾增;二进制 
//以下读写↓ 
//r+;w+;a+;文本ASCII 
//rb+;wb+;ab+二进制 
void writeImage(byte*imgbuf, int size) 
{ 
 //FILE* fp = fopen(shaderFile, "wb"); 
 //由于vs甚么安全性的原因,不让使用fopen,用下面的fopen_s代替; 
 FILE*imgPo; 
 fopen_s(&imgPo, "mag.png", "wb");//这里是用二进制读取,read-r;binary-b;因为只弄r结果出错!!弄了后面那个的再来看这个才发现是这个的问题!! 
 if (imgPo == NULL)return; 
 fwrite(imgbuf, sizeof(char),size,imgPo); 
 fclose(imgPo); 
} 
void readImageFile(const char* Imgname) 
{ 
 //FILE* fp = fopen(shaderFile, "rb"); 
 //由于vs甚么安全性的原因,不让使用fopen,用下面的fopen_s代替; 
 FILE*imgP; 
 fopen_s(&imgP,Imgname,"rb");//这里是用二进制读取,read-r;binary-b;因为只弄r结果出错!!弄了后面那个的再来看这个才发现是这个的问题!! 
 if (imgP == NULL)return ; 
 fseek(imgP, 0L, SEEK_END); 
 long size = ftell(imgP); 
 byte*imgbuf = new byte[size+ 1]; 
 fseek(imgP,0x0L,SEEK_SET);//图片源 
 fread(imgbuf, sizeof(imgbuf[0]), size, imgP); 
 /*for (int j = 0; j < size; j++) 
 cout << (imgbuf[j] & 0xff) << ":";*/ 
 fclose(imgP); 
 
 writeImage(imgbuf, size); 
} 
 
 
//=========================================================== 
 
 
void WriteImage(byte*imgbuf, int size) 
{ 
 
 ofstream imgFo("Image2.png", ios::binary); 
 if (!imgFo) 
 { 
 cerr << "open error!" << endl; 
 abort(); 
 } 
 imgFo.write((char*)imgbuf, size);//一次性写入后面注释的是循环写入 
 
 /* for (int i = 0; i < size; i++) 
 { 
 char ct = (imgbuf[i] & 0xFF); 
 imgFo.write(&ct, sizeof(char)); 
 
 //byte ct = (imgbuf[i] & 0xFF); 
 //imgFo.write((char*)&ct, sizeof(byte)); 
 //尝试这样输出的是否正确. 
 //byte是我自己给名的unsigned char,出来的是对的,用char也可以。都是一个字节。 
 
 }*/ 
 imgFo.close(); 
} 
void ReadImageFile(const char* Imgname) 
{ 
 ifstream imgF(Imgname, ios::binary); 
 if (!imgF) { 
 cerr << "open error!" << endl; 
 abort(); 
 } 
 imgF.seekg(0, ios::end); 
 int size = imgF.tellg(); 
 //查了C++Library Reference才知道怎么得到size。 
 /*int pixscnt; 
 byte width[4], height[4]; 
 
 imgF.seekg(0x10); 
 imgF.read((char*)&width, sizeof(width)); 
 
 imgF.seekg(0x14); 
 imgF.read((char*)&height, sizeof(height)); 
 
 for (int i = 0; i < 4; i++) 
 cout << (width[i] & 0xff) << ":"; 
 
 for (int i = 0; i < 4; i++) 
 cout << (height[i] & 0xff) << ":"; 
 
 pixscnt = (width[2] * (0x100) + width[3])*(height[2] * (0x100) + height[3]); 
 cout << pixscnt << endl;//像素 
 cout << size << endl;*/ 
 byte*imgbuf = new byte[size]; 
 //imgF.seekg(0x10); 
 imgF.seekg(0,ios::beg); 
 imgF.read((char*)imgbuf, size);//一次性读入,书上的不知是错的还是旧的不可行。后面注释的是循环读入 
 /*for (int i = 0; i<size; i++) 
 imgF.read( (char*)&imgbuf[i], sizeof(byte));*/ 
 imgF.close(); 
 /*for (int i = 0; i < size; i++) 
 { 
 cout << hex << (imgbuf[i] & 0xff) << ":"; 
 if (i % 4 == 0)cout << endl; 
 } */ 
WriteImage(imgbuf, size); 
} 
 
int main() 
{ 
 readImageFile("mm.png");//C/C++的 
 ReadImageFile("mm.png");//C++的 
 system("pause"); 
 return 0; 
} 

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。

[!--infotagslink--]

相关文章

  • C++ STL标准库std::vector的使用详解

    vector是表示可以改变大小的数组的序列容器,本文主要介绍了C++STL标准库std::vector的使用详解,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2022-03-06
  • C++中取余运算的实现

    这篇文章主要介绍了C++中取余运算的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-23
  • Python 图片转数组,二进制互转操作

    这篇文章主要介绍了Python 图片转数组,二进制互转操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-09
  • 详解C++ string常用截取字符串方法

    这篇文章主要介绍了C++ string常用截取字符串方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-04-25
  • C++调用C#的DLL程序实现方法

    本文通过例子,讲述了C++调用C#的DLL程序的方法,作出了以下总结,下面就让我们一起来学习吧。...2020-06-25
  • C++中四种加密算法之AES源代码

    本篇文章主要介绍了C++中四种加密算法之AES源代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。...2020-04-25
  • C++ 整数拆分方法详解

    整数拆分,指把一个整数分解成若干个整数的和。本文重点给大家介绍C++ 整数拆分方法详解,非常不错,感兴趣的朋友一起学习吧...2020-04-25
  • C++中 Sort函数详细解析

    这篇文章主要介绍了C++中Sort函数详细解析,sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变...2022-08-18
  • C++万能库头文件在vs中的安装步骤(图文)

    这篇文章主要介绍了C++万能库头文件在vs中的安装步骤(图文),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-23
  • C# 向二进制文件进行读写的操作方法

    该例子使用 BinaryStream 和 BinaryWriter 对二进制文件进行读写操作先上代码再根据我理解的所分享给各位朋友...2020-06-25
  • 详解C++ bitset用法

    这篇文章主要介绍了C++ bitset用法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-04-25
  • 浅谈C++中的string 类型占几个字节

    本篇文章小编并不是为大家讲解string类型的用法,而是讲解我个人比较好奇的问题,就是string 类型占几个字节...2020-04-25
  • C++ Eigen库计算矩阵特征值及特征向量

    这篇文章主要为大家详细介绍了C++ Eigen库计算矩阵特征值及特征向量,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-04-25
  • C++ pair的用法实例详解

    这篇文章主要介绍了C++ pair的用法实例详解的相关资料,需要的朋友可以参考下...2020-04-25
  • VSCode C++多文件编译的简单使用方法

    这篇文章主要介绍了VSCode C++多文件编译的简单使用方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-03-29
  • C++中的循环引用

    虽然C++11引入了智能指针的,但是开发人员在与内存的斗争问题上并没有解放,如果我门实用不当仍然有内存泄漏问题,其中智能指针的循环引用缺陷是最大的问题。下面通过实例代码给大家介绍c++中的循环引用,一起看看吧...2020-04-25
  • C++随机点名生成器实例代码(老师们的福音!)

    这篇文章主要给大家介绍了关于C++随机点名生成器的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-04-25
  • C++如何删除map容器中指定值的元素详解

    map容器是C++ STL中的重要一员,删除map容器中value为指定元素的问题是我们经常与遇到的一个问题,下面这篇文章主要给大家介绍了关于利用C++如何删除map容器中指定值的元素的相关资料,需要的朋友可以参考借鉴,下面来一起看看吧。...2020-04-25
  • 整理C# 二进制,十进制,十六进制 互转

    c#下进制互转代码...2020-06-25
  • C++ 约瑟夫环问题案例详解

    这篇文章主要介绍了C++ 约瑟夫环问题案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下...2021-08-15