1

我需要帮助以 C/C++ 保存 PPM 文件。我想从三个基本颜色通道中保存 PPM 图像。通道表示为 void * 指针(pRed、pGreen 和 pBlue)。这些值是固定的,所以我无法更改它们。

还有一个变量来决定如何解释 void 指针:这个变量是:

bpp=1 正确的类型是 unsigned char *。缓冲区元素的类型为 unsigned char

bpp=2 正确的类型是 unsigned short *。缓冲区元素是 unsigned short 类型。

现在我将从这三个颜色通道中保存一个 ppm 文件。

关于 bpp 变量的声明是:

//bpp=1:
unsigned char* dRed = (unsigned char*)pImg->pRed;
unsigned char* dGreen = (unsigned char*)pImg->pGreen;
unsigned char* dBlue = (unsigned char*)pImg->pBlue;
unsigned char* dBw = (unsigned char*)pImg->pBw;

//bpp=2:
unsigned short* dRed = (unsigned short*)pImg->pRed;
unsigned short* dGreen = (unsigned short*)pImg->pGreen;
unsigned short* dBlue = (unsigned short*)pImg->pBlue;
unsigned short* dBw = (unsigned short*)pImg->pBw;

我编写 ppm 文件的代码:

ofstream output(fname, ios::binary|ios::out);

output << "P3"<< endl <<"# foreground "<<endl;
output << itoa(width, fname, 10);
output << " ";
output << itoa(height, fname, 10);
output << endl;
output << itoa(255, fname, 10) << endl;
...     
for(int i=0; i<(height*width); i++){
  if(bytesPerP==1){
    output << (unsigned char)((char*)dRed) << " ";
    output << (unsigned char)((char*)dGreen) << " ";
    output << (unsigned char)((char*)dBlue) << " ";
    dRed = dRed + (i * bytesPerP);
    dGreen = dGreen + (i * bytesPerP);
    dBlue = dBlue + (i * bytesPerP);
}else if(bytesPerP==2){
    output << ((unsigned short)((char*)dRed))%256 << " ";
    output << (unsigned short)((char*)dGreen)%256 << " ";
    output << (unsigned short)((char*)dBlue)%256 << " ";

    dRed = dRed + (i * bytesPerP);
    dGreen = dGreen + (i * bytesPerP);
    dBlue = dBlue + (i * bytesPerP);
}

这是错误的任何地方,因为我得到了记录图像的正确“结构”,但颜色完全奇怪(迷幻风格):) 我在互联网上搜索并测试了很多东西,但没有一个有效。有人能指出我正确的方向吗?

4

1 回答 1

1

问题是您如何从图像缓冲区输出字节......您需要取消引用指针以获取实际的数据信息,而不是输出实际的指针值本身。

例如,这个:

output << (unsigned char)((char*)dRed) << " ";

输出 4 或 8 字节指针值dRed强制转换为unsigned char... 你想要指针指向dRed的值,可以这样完成:

output << *dRed << " ";

其次,如果每个颜色通道的数据存储在单独的连续线性阵列中(即,红色仅指向红色像素,绿色仅指向绿色像素,而缓冲区是红色像素,则为绿色-pixel,然后是蓝色像素),那么在每通道 16 位的情况下,您没有正确增加值。如果缓冲区是线性数组,您可以简单地执行以下操作:

 output << " " << dRed[i] << " " << dGreen[i] << " " << dBlue[i] << endl;

其中额外的空白是为了符合 PPM 标准,其中像素值样本应该被空白字符包围。

最后,在您输出的“普通”PPM 格式中,任何行都不应超过 70 个字符......你肯定会超过这个限制......

于 2012-02-14T18:43:59.330 回答