问题读取 ppm p6 图像完成。 C++
Issues reading ppm p6 image to completion. c++
int main(){
std::fstream myfile; // file object is created
myfile.open ("green.ppm");
std::string line;
unsigned red,green,blue; //to output values on 0 -255 scale.
int width, height = 0;
if (myfile.is_open())
{
std::getline (myfile,line); //type of file, skip, it will always be for this code p6
std::getline (myfile,line); // width and height of the image
std::stringstream lineStream(line); //extract the width and height;
lineStream >> width;
lineStream >> height;
// std::cout<< width << " " << height <<" \n";
getline (myfile,line); //skip magic number
getline (myfile,line); // reach the matrix of numbers
for (int i = 0; i<(width*height*3) ; i= i+3){
char num = line[i]; uint8_t number = num; red = number;
num = line[i+1]; number = num; green = number;
num = line[i+2]; number = num; blue = number;
std::cout<<"pixel " << i/3 << " is " << red << " " << green << " " << blue << std::endl;
}
//char to uint_8t to unsigned is a basic an inefficient way I found that takes the pixel rgb values in my ppm file and allows me to interpret them from a range of 0-255
}
// cout<<counter<<endl;
myfile.close();
return 0;
}
当我在不同的 ppm 图像上 运行 这段代码时,它实际上确实正确地提取了 rgb 值,但问题是它并没有完全做到这一点。一个基本的 800 x 800 图像有 640000 个像素,这段代码读取大约 40800,然后结束,就好像它不再存在一样。
我认为这是对 ppm 格式本身的误解。我认为在 header 格式之外,即具有类型、宽度和大小以及幻数的文件,只有一行,没有更多的 '\n' 字符。因此矩阵可以被读取为连续的字符数组。
为什么这个程序停在这么奇怪的地方?
我不擅长 C++,但我会尝试解释这里的错误。
如果文件类型为 p6
,则图像数据以字节格式存储,每个颜色分量 (r,g,b) 一个字节。从代码中的注释看来,您总是期望 p6
。这意味着矩阵将像您假设的那样存储为一个连续的数据块。其他选项是 p3
,它以 ASCII 格式存储颜色分量(例如 0 0 0
)。
问题是在 p6
类型的文件中,数据块是二进制的,而您将其视为字符串。考虑一下,r,g,b = (65, 13, 10) 的像素将像这样以二进制编码:
0x41, 0xD, 0xA
getline
将读取评估为 ASCII 字符 A
的第一个字节,但会在其后停止读取,因为 0xD0xA
是 \r\n
,一个换行符(在 Windows) 这是 getline
.
的分隔符
而不是:
getline (myfile,line); // reach the matrix of numbers
我会做类似的事情:
int bufsz = 3*width*height;
char* pixelData = new char[bufsz];
myfile.read(pixelData,bufsz);
for ( int i = 0; i < bufsz; i+=3 ) {
unsigned char red = pixelData[i];
unsigned char green = pixelData[i+1];
unsigned char blue = pixelData[i+2];
// use unsigned char to express range [0,255],
// this may make compiler to issue warnings
// but it should be safe to do a cast
}
// don't forget to release pixelData when done with it,
// maybe use smart pointers
delete[] pixelData;
注意: 矩阵块之前的行包含颜色分量的最大值,PPM 声明它可以大于 255,因此在这些情况下您需要小心因为 char
数组不够好,您至少需要 short
数组(2 个字节)
有用的链接:
int main(){
std::fstream myfile; // file object is created
myfile.open ("green.ppm");
std::string line;
unsigned red,green,blue; //to output values on 0 -255 scale.
int width, height = 0;
if (myfile.is_open())
{
std::getline (myfile,line); //type of file, skip, it will always be for this code p6
std::getline (myfile,line); // width and height of the image
std::stringstream lineStream(line); //extract the width and height;
lineStream >> width;
lineStream >> height;
// std::cout<< width << " " << height <<" \n";
getline (myfile,line); //skip magic number
getline (myfile,line); // reach the matrix of numbers
for (int i = 0; i<(width*height*3) ; i= i+3){
char num = line[i]; uint8_t number = num; red = number;
num = line[i+1]; number = num; green = number;
num = line[i+2]; number = num; blue = number;
std::cout<<"pixel " << i/3 << " is " << red << " " << green << " " << blue << std::endl;
}
//char to uint_8t to unsigned is a basic an inefficient way I found that takes the pixel rgb values in my ppm file and allows me to interpret them from a range of 0-255
}
// cout<<counter<<endl;
myfile.close();
return 0;
}
当我在不同的 ppm 图像上 运行 这段代码时,它实际上确实正确地提取了 rgb 值,但问题是它并没有完全做到这一点。一个基本的 800 x 800 图像有 640000 个像素,这段代码读取大约 40800,然后结束,就好像它不再存在一样。
我认为这是对 ppm 格式本身的误解。我认为在 header 格式之外,即具有类型、宽度和大小以及幻数的文件,只有一行,没有更多的 '\n' 字符。因此矩阵可以被读取为连续的字符数组。
为什么这个程序停在这么奇怪的地方?
我不擅长 C++,但我会尝试解释这里的错误。
如果文件类型为 p6
,则图像数据以字节格式存储,每个颜色分量 (r,g,b) 一个字节。从代码中的注释看来,您总是期望 p6
。这意味着矩阵将像您假设的那样存储为一个连续的数据块。其他选项是 p3
,它以 ASCII 格式存储颜色分量(例如 0 0 0
)。
问题是在 p6
类型的文件中,数据块是二进制的,而您将其视为字符串。考虑一下,r,g,b = (65, 13, 10) 的像素将像这样以二进制编码:
0x41, 0xD, 0xA
getline
将读取评估为 ASCII 字符 A
的第一个字节,但会在其后停止读取,因为 0xD0xA
是 \r\n
,一个换行符(在 Windows) 这是 getline
.
而不是:
getline (myfile,line); // reach the matrix of numbers
我会做类似的事情:
int bufsz = 3*width*height;
char* pixelData = new char[bufsz];
myfile.read(pixelData,bufsz);
for ( int i = 0; i < bufsz; i+=3 ) {
unsigned char red = pixelData[i];
unsigned char green = pixelData[i+1];
unsigned char blue = pixelData[i+2];
// use unsigned char to express range [0,255],
// this may make compiler to issue warnings
// but it should be safe to do a cast
}
// don't forget to release pixelData when done with it,
// maybe use smart pointers
delete[] pixelData;
注意: 矩阵块之前的行包含颜色分量的最大值,PPM 声明它可以大于 255,因此在这些情况下您需要小心因为 char
数组不够好,您至少需要 short
数组(2 个字节)
有用的链接: