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

相关文章

SpringBoot集成Milvus实现数据增删改查功能

《SpringBoot集成Milvus实现数据增删改查功能》milvus支持的语言比较多,支持python,Java,Go,node等开发语言,本文主要介绍如何使用Java语言,采用springboo... 目录1、Milvus基本概念2、添加maven依赖3、配置yml文件4、创建MilvusClient

SpringValidation数据校验之约束注解与分组校验方式

《SpringValidation数据校验之约束注解与分组校验方式》本文将深入探讨SpringValidation的核心功能,帮助开发者掌握约束注解的使用技巧和分组校验的高级应用,从而构建更加健壮和可... 目录引言一、Spring Validation基础架构1.1 jsR-380标准与Spring整合1

OpenCV图像形态学的实现

《OpenCV图像形态学的实现》本文主要介绍了OpenCV图像形态学的实现,包括腐蚀、膨胀、开运算、闭运算、梯度运算、顶帽运算和黑帽运算,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起... 目录一、图像形态学简介二、腐蚀(Erosion)1. 原理2. OpenCV 实现三、膨胀China编程(

MySQL 中查询 VARCHAR 类型 JSON 数据的问题记录

《MySQL中查询VARCHAR类型JSON数据的问题记录》在数据库设计中,有时我们会将JSON数据存储在VARCHAR或TEXT类型字段中,本文将详细介绍如何在MySQL中有效查询存储为V... 目录一、问题背景二、mysql jsON 函数2.1 常用 JSON 函数三、查询示例3.1 基本查询3.2

SpringBatch数据写入实现

《SpringBatch数据写入实现》SpringBatch通过ItemWriter接口及其丰富的实现,提供了强大的数据写入能力,本文主要介绍了SpringBatch数据写入实现,具有一定的参考价值,... 目录python引言一、ItemWriter核心概念二、数据库写入实现三、文件写入实现四、多目标写入

使用Python将JSON,XML和YAML数据写入Excel文件

《使用Python将JSON,XML和YAML数据写入Excel文件》JSON、XML和YAML作为主流结构化数据格式,因其层次化表达能力和跨平台兼容性,已成为系统间数据交换的通用载体,本文将介绍如何... 目录如何使用python写入数据到Excel工作表用Python导入jsON数据到Excel工作表用

Mysql如何将数据按照年月分组的统计

《Mysql如何将数据按照年月分组的统计》:本文主要介绍Mysql如何将数据按照年月分组的统计方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql将数据按照年月分组的统计要的效果方案总结Mysql将数据按照年月分组的统计要的效果方案① 使用 DA

鸿蒙中Axios数据请求的封装和配置方法

《鸿蒙中Axios数据请求的封装和配置方法》:本文主要介绍鸿蒙中Axios数据请求的封装和配置方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.配置权限 应用级权限和系统级权限2.配置网络请求的代码3.下载在Entry中 下载AxIOS4.封装Htt

使用Python实现一键隐藏屏幕并锁定输入

《使用Python实现一键隐藏屏幕并锁定输入》本文主要介绍了使用Python编写一个一键隐藏屏幕并锁定输入的黑科技程序,能够在指定热键触发后立即遮挡屏幕,并禁止一切键盘鼠标输入,这样就再也不用担心自己... 目录1. 概述2. 功能亮点3.代码实现4.使用方法5. 展示效果6. 代码优化与拓展7. 总结1.

Python获取中国节假日数据记录入JSON文件

《Python获取中国节假日数据记录入JSON文件》项目系统内置的日历应用为了提升用户体验,特别设置了在调休日期显示“休”的UI图标功能,那么问题是这些调休数据从哪里来呢?我尝试一种更为智能的方法:P... 目录节假日数据获取存入jsON文件节假日数据读取封装完整代码项目系统内置的日历应用为了提升用户体验,