【问题标题】:Seperate 4 channels (R,G,G,B) of .raw image file and save them as valid image in c++分离 .raw 图像文件的 4 个通道(R、G、G、B)并将它们保存为 C++ 中的有效图像
【发布时间】:2019-07-12 20:49:42
【问题描述】:

我必须获取一个 .raw12 文件,分离 4 个通道(R、G、G、B)并将它们保存为有效图像(内存中的 8 位),而不使用 C++ 中的任何外部库 (.ppm)。

您可以阅读有关 raw12 here 和 ppm 文件格式 here 的信息。

我已经编写了代码,但它给了我这个输出。

Click 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();
}

Github Link To The Code

【问题讨论】:

  • 检查你的逻辑: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


【解决方案1】:

我已经大大简化了您的代码并使其仅输出一个通道,因为您的问题是您应该为每个通道生成一张图像。

现在你有了一些工作,你可以添加回其他部分 - 从工作开始更容易!我不会为你做所有的挑战......那会没有乐趣,让你没有任何成就感。您可以完成剩下的工作 - 祝您好运!

#include <fstream>
#include <iostream>
using namespace std;

// Enough for one line of the input image
const int BUFFERSIZE = 4096 * 3;

int main(){

    ifstream infile;
    ofstream outfile;

    infile.open("portrait.raw12", ios::binary | ios::in);
    outfile.open("result.pgm", ios::binary);

    // Write single channel PGM file
    outfile << "P5\n2048 1536\n255\n";

    unsigned char * bufferRow = new unsigned char[BUFFERSIZE];

    if(!infile)
    {
        cout<<"Failed to open"<<endl;
    }
    int size=2048*1536;
    unsigned char * RedChannel=new unsigned char[size];
    unsigned char * Redp = RedChannel;

    for(int rowNum=0;rowNum<1536;rowNum++){
        // Read an entire row
        infile.read(reinterpret_cast<char*>(bufferRow), BUFFERSIZE);
        if(rowNum%2==0){
            for(int i=0;i<BUFFERSIZE;i+=3){
                *Redp++=bufferRow[i];
            }
        }
    }
    infile.close();
    outfile.write(reinterpret_cast<char*>(RedChannel), size);
    outfile.close();
}

【讨论】:

  • 非常感谢先生。你太酷了。 o。我只提出了一些琐碎的问题。 Q1。为什么第 29 行中的 for 循环是“rowNum
  • 我很高兴它有帮助。 Q1。我发现 RAW12 格式的描述真的很差,并且花了很长时间试图理解它的含义并使文件大小与行、列和样本大小相匹配,而 1536 似乎是正确的。 Q2 我认为*Redp++ = xyz 更容易正确,因为数据已写入并且指针会一次性更新,而如果您这样做Red[j] = xyz; ...; ...; j +=1,您很容易不同步而不更新指针。
  • 至于好,程序还有很多可以改进的地方,但我们都在学习!
  • 非常感谢,先生。当我在 stackoverflow 中发布这个问题时,我几乎不希望有人会为我解决这个问题。真的,先生,非常感谢您宝贵的时间和精力。
猜你喜欢
  • 1970-01-01
  • 2018-05-09
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-05-11
  • 1970-01-01
相关资源
最近更新 更多