【发布时间】:2019-07-12 20:49:42
【问题描述】:
我必须获取一个 .raw12 文件,分离 4 个通道(R、G、G、B)并将它们保存为有效图像(内存中的 8 位),而不使用 C++ 中的任何外部库 (.ppm)。
您可以阅读有关 raw12 here 和 ppm 文件格式 here 的信息。
我已经编写了代码,但它给了我这个输出。
我尝试了很多东西,但它总是给我类似于上面的输出。我认为数据类型转换有问题,但我不确定。
我尝试从 2 天开始调试它,仍然没有运气。
这是这段代码。
#include <fstream>
#include <iostream>
using namespace std;
const int BUFFERSIZE = 4096;
int main ()
{
ifstream infile;
infile.open("portrait.raw12", ios::binary | ios::in);
ofstream outfile;
outfile.open("Redtwo.ppm", ios::binary);
//outfile.write("P6 ", 3);
//outfile.write("1536 2048 ", 8);
//outfile.write("2048 ", 4);
//outfile.write("255 ", 4);
//outfile << "P6\n" << 1536 << "\n" << 2048 << "\n255\n";
outfile << "P6" << "\n"
<< 1536 << " "
<< 2048 << "\n"
<< 255 << "\n"
;
uint8_t * bufferRow = new uint8_t[BUFFERSIZE];
if(!infile)
{
cout<<"Failed to open"<<endl;
}
int size=1536*2048*3;
char * RedChannel=new char[size];
int GreenChannel_1,GreenChannel_2,BlueChannel;
int rowNum=0;
int i=0;
int j=0;
int pixel=1;
while(rowNum<3072)
{
infile.read(reinterpret_cast<char*>(bufferRow), BUFFERSIZE);
if(rowNum%2==0)
{
while(i<BUFFERSIZE)
{
RedChannel[j]=(uint8_t)bufferRow[i];
GreenChannel_1=((uint8_t)(bufferRow[i+1] & 0x0F) << 4) | ((uint8_t)(bufferRow[i+2] >> 4) & 0x0F);
i+=3;
//Collect s;
//s.r=(char)RedChannel[j];
//s.g=(char)0;
//s.b=(char)0;
//unsigned char c = (unsigned char)(255.0f * (float)RedChannel[j] + 0.5f);
//outfile.write((char*) &c, 3);
//outfile.write((char*) 255, sizeof(c));
//outfile.write(reinterpret_cast<char*>(&RedChannel), 4);
if(pixel<=3 && rowNum<5)
{
cout<<"RedChannel: "<<RedChannel[j]<<endl;
if(pixel!=3)
cout<<"GreenChannel 1: "<<GreenChannel_1<<endl;
}
pixel++;
j++;
}
RedChannel[j]='\n';
j++;
}
else
{
while(i<BUFFERSIZE)
{
GreenChannel_2=(uint8_t)bufferRow[i];
BlueChannel=((uint8_t)(bufferRow[i+1] & 0x0F) << 4) | ((uint8_t)(bufferRow[i+2] >> 4) & 0x0F);
i+=3;
if(pixel<=3 && rowNum<5)
{
cout<<"GreenChannel 2: "<<GreenChannel_2<<endl;
if(pixel!=3)
cout<<"BlueChannel: "<<BlueChannel<<endl;
}
pixel++;
}
}
rowNum++;
i=0;
pixel=1;
if(rowNum<5)
cout<<" "<<endl;
}
infile.close();
outfile.write(RedChannel, size);
outfile.close();
}
【问题讨论】:
-
检查你的逻辑:RGGB 每 6 个字节对齐一次,RedChannel 应该有 12 位,而你只是得到 8 位。您链接的页面底部的示例为您提供了一个很好的示例。格式似乎不太好。我真的很怀念对值的解释(原始应该是线性的)。
-
请提供raw12数据文件。
-
您链接到的文档说图像高 3072 行,宽 4096 列,但您编写的 PPM 格式声称有 1536 列宽 x 2048 行高,因此您将大小减半并旋转了 90 度。为什么?
-
您的数据是每个样本 12 位,但您在 PPM 标头中写入了 255 的
MAX_VALUE,这意味着您的 PPM 文件每个样本有 8 位。我本来希望您在标头中写入 4096 的MAX_VALUE,然后每个样本写入 2 个字节。 -
@GiacomoCatenazzi 我真的很抱歉。在给出的问题陈述中,我必须取 8 位并将这 8 位写入 .ppm 文件。我必须将 12 位转换为 8 位,为此我必须砍掉 4 个最低有效位。在问题陈述中还提到我必须打印前 5x5 像素的强度值。为了避免混淆,我应该提到这一点。
标签: c++ image-processing rgb ppm raw-file