本机系统配置:ThinkPadT570、Windows10、QT5.12.2(QtCreater4.8.2)
由于工作需要,我现在要解析Mono 12 Packed Format格式保存的图像数据,自己摸索了半天外加请教了同事,算是弄清楚这个格式怎么存的了:
假设保存的文件一共3个字节,按照16进制显示如下:
01 01 01 //地位在前,高位在后
换算成二进制后如下:
0001 0000 0001 0000 0001 0000 //低位在前,高位在后
那么,前12位表示一个像素的值,后12位表示一个像素的值,像素值范围是(0~4095)
如果我们将其转换成两个unsigned short的话,对应关系见下图:
具体代码如下:
首先是将12位压缩格式解析成16位的代码:
QFileDialog *fileDlg = new QFileDialog(this);
fileDlg->setWindowTitle("Choose Pictures");
QStringList qstrFilters;
qstrFilters<<"Image files(*.bmp *.jpg *.pbm *.pgm *.png *.ppm *.xbm *.xpm)";
qstrFilters<<"Any files (*)";
fileDlg->setNameFilters(qstrFilters);//设置文件过滤器
fileDlg->setFileMode(QFileDialog::ExistingFile);//设置能选择多个文件,如果是单个文件就写成QFileDialog::ExistingFile
if(fileDlg->exec() == QDialog::Accepted)
{
QStringList strPathList = fileDlg->selectedFiles();
QFile file(strPathList.at(0));
if(!file.open(QIODevice::ReadOnly))
{
QMessageBox mesg;
mesg.warning(this,"警告","打开图片失败!");
}
QByteArray t = file.readAll();//读取文件中所有数据,存放到t中
file.close();
int nFileSize = t.size();
int nNum = nFileSize%3;//数据按照每3个字节进行一次转换,此处计算循环的次数
if(nNum!=0)
{
nNum = nFileSize/3+1;
}
else
{
nNum = nFileSize/3;
}
QVector<unsigned short> vShortData;
for(int i=0;i<nNum;i++)//数据按照每3个字节进行一次转换
{
unsigned char c1,c2,c3;//从内存里读取到的3个字节数据
unsigned char n8bit;//转换时候临时使用
unsigned short n16bit;//转换时候临时使用
unsigned short n1,n2;//转换后的数据
memcpy((char*)&c1,(char*)(t.data()+3*i),1*sizeof(char));
memcpy((char*)&c2,(char*)(t.data()+3*i+1),1*sizeof(char));
//memcpy((char*)&c3,(char*)(t.data()+3*i+2),1*sizeof(char));
//转换二进制数据中第一个字节
n16bit = c1;
n16bit = n16bit<<4;
n1 = n16bit;
//转换二进制数据中第二个字节的前4位
n8bit = c2<<4;
n8bit = n8bit>>4;
n16bit = n8bit;
n1 = n1+n16bit;
//转换二进制数据中第二个字节的后4位和第三个字节
memcpy((char*)&n2,(char*)(t.data()+3*i+1),2*sizeof(char));
n2 = n2>>4;
vShortData.append(n1);//将转换后的数据存到vShortData中
vShortData.append(n2);
}
}
else
{
QMessageBox mesg;
mesg.warning(this,"警告","打开图片失败!");
}
fileDlg->close();
delete fileDlg;
fileDlg = nullptr;
接下来是逆向的,将16位压缩成12位的代码:
void MainWindow::Translate16to12(QVector<unsigned short> &vData16,QVector<unsigned char> &vData12)
{
if(vData16.isEmpty())
{
return;
}
int nNum12 = vData16.size()/2*3;
vData12.resize(nNum12);
int nCircleTime = vData16.size()/2;
for (int i=0;i<nCircleTime;i++)
{
unsigned short n1,n2;//16位的数据
unsigned char c1,c2,c3;//
n1 = vData16.at(i*2);
n2 = vData16.at(i*2+1);
c1 = n1>>4;//获取c1的值
n1 = n1<<12;
n1 = n1>>12;
c3 = n2>>4;//获取c3的值
n2 = n2<<12;
n2 = n2>>8;
c2 = n2+n1;//获取c2的值
memcpy((char*)&vData12[3*i],(char*)&c1,sizeof(char));
memcpy((char*)&vData12[3*i+1],(char*)&c2,sizeof(char));
memcpy((char*)&vData12[3*i+2],(char*)&c3,sizeof(char));
}
//start write file :把数据写到本地,便于与原来的12位深数据进行对比
QByteArray ba;
ba.resize(vData12.size());
memcpy((char*)ba.data(),(char*)&vData12[0],ba.size()*sizeof(char));//将之前的代码处理后得到的QVector<unsigned short>类型的变量vShortData中的数据拷贝到 QByteArray ba;中
QFile writefile("2.dat");//如果文件不存在,则创建该文件。路径在执行文件的上一级路径下。
writefile.open(QIODevice::ReadWrite);
QDataStream out(&writefile);
writefile.write(ba.data(),ba.size());//这种方式也不会有多余字节
writefile.close();
}