Mono 12 Packed Format单个像素12位深灰度图压缩格式读取解析(一般来说只有工业相机会用到这种格式)

本机系统配置: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();
}

猜你喜欢

转载自blog.csdn.net/weixin_43935474/article/details/90372825