LSB信息隐藏之BMP图像数据的读取

2023-10-31 11:32

本文主要是介绍LSB信息隐藏之BMP图像数据的读取,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

#include "stdafx.h"
#include "LSB_Coder.h"


//-------------------------------------------------------------------------------------------
//读图像的位图数据、宽、高、颜色表及每像素位数等数据进内存,存放在相应的全局变量中
bool LSB::readBmp(char *bmpName) 
{
FILE *fp=fopen(bmpName,"rb");//二进制读方式打开指定的图像文件


if(fp==0) return 0;


//跳过位图文件头结构BITMAPFILEHEADER


fseek(fp, sizeof(BITMAPFILEHEADER),0);


//定义位图信息头结构变量,读取位图信息头进内存,存放在变量head中


BITMAPINFOHEADER head;  


fread(&head, sizeof(BITMAPINFOHEADER), 1,fp); //获取图像宽、高、每像素所占位数等信息


bmpWidth = head.biWidth;


bmpHeight = head.biHeight;


biBitCount = head.biBitCount;//定义变量,计算图像每行像素所占的字节数(必须是4的倍数)


int lineByte=(bmpWidth * biBitCount/8+3)/4*4;//灰度图像有颜色表,且颜色表表项为256


if(biBitCount==8)
{


//申请颜色表所需要的空间,读颜色表进内存


pColorTable=new RGBQUAD[256];


fread(pColorTable,sizeof(RGBQUAD),256,fp);


}


//申请位图数据所需要的空间,读位图数据进内存


pBmpBuf= new unsigned char[lineByte * bmpHeight];


fread(pBmpBuf,1,lineByte * bmpHeight,fp);


fclose(fp);//关闭文件


return 1;//读取文件成功
}


//-----------------------------------------------------------------------------------------
//给定一个图像位图数据、宽、高、颜色表指针及每像素所占的位数等信息,将其写到指定文件中
//bool LSB::saveBmp(char *bmpName, unsigned char *imgBuf, int width, int height, int biBitCount, RGBQUAD *pColorTable)
bool LSB::saveBmp(char *bmpName)
{


//如果位图数据指针为0,则没有数据传入,函数返回


if(!pBmpBuf)


return 0;


//颜色表大小,以字节为单位,灰度图像颜色表为1024字节,彩色图像颜色表大小为0


int colorTablesize=0;


if(biBitCount==8)


colorTablesize=1024;


//待存储图像数据每行字节数为4的倍数


int lineByte=(bmpWidth * biBitCount/8+3)/4*4;

//以二进制写的方式打开文件


FILE *fp=fopen(bmpName,"wb");


if(fp==0) return 0;


//申请位图文件头结构变量,填写文件头信息


BITMAPFILEHEADER fileHead;


fileHead.bfType = 0x4D42;//bmp类型


//bfSize是图像文件4个组成部分之和


fileHead.bfSize= sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)


+ colorTablesize + lineByte*bmpHeight;


fileHead.bfReserved1 = 0;


fileHead.bfReserved2 = 0;


//bfOffBits是图像文件前3个部分所需空间之和


fileHead.bfOffBits=54+colorTablesize;


//写文件头进文件


fwrite(&fileHead, sizeof(BITMAPFILEHEADER),1, fp);


//申请位图信息头结构变量,填写信息头信息


BITMAPINFOHEADER head; 


head.biBitCount=biBitCount;


head.biClrImportant=0;


head.biClrUsed=0;


head.biCompression=0;


head.biHeight=bmpHeight;


head.biPlanes=1;


head.biSize=40;


head.biSizeImage=lineByte*bmpHeight;


head.biWidth=bmpWidth;


head.biXPelsPerMeter=0;


head.biYPelsPerMeter=0;


//写位图信息头进内存


fwrite(&head, sizeof(BITMAPINFOHEADER),1, fp);


//如果灰度图像,有颜色表,写入文件 


if(biBitCount==8)


fwrite(pColorTable, sizeof(RGBQUAD),256, fp);


//写位图数据进文件


fwrite(pBmpBuf, bmpHeight*lineByte, 1, fp);


//关闭文件


fclose(fp);


    //清除缓冲区,pBmpBuf和pColorTable是全局变量,在文件读入时申请的空间
//delete []pBmpBuf;
//if(biBitCount==8)
//delete []pColorTable;


return 1;


}




//-------------------------------------------------------------------------------------------
//一下为信息隐藏部分
void LSB::LSBCoder(const char*textFileName)
{
ifstream textFile;
textFile.open(textFileName,ios::in | ios::binary);
//_infoStream.open(_originalPicPath,ios::in | ios::binary);
textFile.seekg(0,textFile.end);
long textFileLength = textFile.tellg();


BYTE* pTextFile = new BYTE[textFileLength+1];
cout<<"隐藏时文件长度:"<<textFileLength<<endl;
textFile.seekg(0,textFile.beg);
textFile.read((char*)pTextFile,textFileLength);
textFile.close();


BYTE textData;
for(int i=0,k=0; i< textFileLength; ++i)
{
for(int j=0; j<8; ++j)
{
textData = pTextFile[i]>>j;
textData = textData&0x01;
if(textData==0)
{
 pBmpBuf[k+32] = pBmpBuf[k+32]&0xfe;
}
else
{
pBmpBuf[k+32] = pBmpBuf[k+32]|0x01;
}
++k;
}
}
cout<<"信息隐藏完毕"<<endl;
long length;
for(int i=0; i<32; ++i)
{
length = textFileLength>>i;
length = length&0x00000001;
if(length==0)
{
pBmpBuf[i] = pBmpBuf[i]&0x1e;
}
else
{
pBmpBuf[i] =pBmpBuf[i]|0x01;
}


}
}
//------------------------------------------------
//  解码基于LSB的信息隐藏
//-------------------------------------------------
void  LSB::LSBDecoder(const char* textFileName)
{
long length = 0x00000000;
BYTE bit;
//获取txt文件长度
for(int i=0; i<32; ++i)
{
bit =pBmpBuf[i]&0x01;
 if(bit==0)
 {
  length = length&0x7fffffff;
 }
 else
 {
 length = length|0x80000000;
 }
  if (i<31)    length = length>>1;
}
cout<<"解码时文件长度:"<<length<<endl;
//开始解码
BYTE* pTextFile = new BYTE[length];
BYTE textData;
for(int i=0,k=0; i<length*8; ++i)
{
if(i && i%8==0){++k;}
textData = pBmpBuf[i+32]&0x01;
if(textData==0)
{
 pTextFile[k] = pTextFile[k]&0x7f;
}
else
{
 pTextFile[k] = pTextFile[k]|0x80;
}
if (i%8 != 7) pTextFile[k] = pTextFile[k]>>1;
}
cout<<"解码完毕"<<endl;


 ofstream textFile;
 textFile.open(textFileName,ios::out | ios::binary);
 textFile.write((char*)pTextFile,length);
 textFile.close();
 delete pTextFile;
 //char savepicPath[]="Apple_Coder.bmp";
 //saveBmp(savepicPath, pBmpBuf, bmpWidth, bmpHeight, biBitCount, pColorTable);
}

这篇关于LSB信息隐藏之BMP图像数据的读取的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/315064

相关文章

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

Python将大量遥感数据的值缩放指定倍数的方法(推荐)

《Python将大量遥感数据的值缩放指定倍数的方法(推荐)》本文介绍基于Python中的gdal模块,批量读取大量多波段遥感影像文件,分别对各波段数据加以数值处理,并将所得处理后数据保存为新的遥感影像... 本文介绍基于python中的gdal模块,批量读取大量多波段遥感影像文件,分别对各波段数据加以数值处

使用MongoDB进行数据存储的操作流程

《使用MongoDB进行数据存储的操作流程》在现代应用开发中,数据存储是一个至关重要的部分,随着数据量的增大和复杂性的增加,传统的关系型数据库有时难以应对高并发和大数据量的处理需求,MongoDB作为... 目录什么是MongoDB?MongoDB的优势使用MongoDB进行数据存储1. 安装MongoDB

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

SpringBoot使用Apache Tika检测敏感信息

《SpringBoot使用ApacheTika检测敏感信息》ApacheTika是一个功能强大的内容分析工具,它能够从多种文件格式中提取文本、元数据以及其他结构化信息,下面我们来看看如何使用Ap... 目录Tika 主要特性1. 多格式支持2. 自动文件类型检测3. 文本和元数据提取4. 支持 OCR(光学

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

Linux使用dd命令来复制和转换数据的操作方法

《Linux使用dd命令来复制和转换数据的操作方法》Linux中的dd命令是一个功能强大的数据复制和转换实用程序,它以较低级别运行,通常用于创建可启动的USB驱动器、克隆磁盘和生成随机数据等任务,本文... 目录简介功能和能力语法常用选项示例用法基础用法创建可启动www.chinasem.cn的 USB 驱动

C#实现获取电脑中的端口号和硬件信息

《C#实现获取电脑中的端口号和硬件信息》这篇文章主要为大家详细介绍了C#实现获取电脑中的端口号和硬件信息的相关方法,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 我们经常在使用一个串口软件的时候,发现软件中的端口号并不是普通的COM1,而是带有硬件信息的。那么如果我们使用C#编写软件时候,如

Oracle数据库使用 listagg去重删除重复数据的方法汇总

《Oracle数据库使用listagg去重删除重复数据的方法汇总》文章介绍了在Oracle数据库中使用LISTAGG和XMLAGG函数进行字符串聚合并去重的方法,包括去重聚合、使用XML解析和CLO... 目录案例表第一种:使用wm_concat() + distinct去重聚合第二种:使用listagg,