我正在努力像素化由存储在二进制(P6) PPM 文件中的 RGB 值组成的图像。像素化图像的步骤如下:
- 读入二进制数据并将其存储在一维数组中
- 在大小为“c”(行 x 列)的单元格中迭代存储在此数组中的数据。这个变量“c”可以由用户更改,但对于这个程序,它当前设置为
int c = 4;
意味着它迭代块尺寸的像素数据4 x 4
- 现在找到每个大小为“c”的单元格中的平均颜色值
- 将每个平均颜色值输出到新的输出 PPM 文件
每个二进制 PPM 文件都以以下格式的标题开头,由“幻数”组成,然后是宽度、高度,最后是最大颜色值 255。标题注释被忽略。以下标题示例显示格式为 P6(因此为二进制文件)的 PPM 图像,宽度为 16,高度为 16,最大颜色值为 255:
P6
16
16
255
我在哪里挣扎:
- 我不确定如何找到每个单元格的平均 RGB 值,然后将其输出到新的数据流。在线性读取数据(即不将其读取到数组(或缓冲区)中)时,我想出了一种方法来执行此操作,方法是将该单元格中的总颜色除以循环迭代次数,但事实证明这是不正确的。
- 嵌套的 for 循环是否以改变输出图像方向的方式排序,例如它是否将其旋转 180 度等?我正在努力通过读取原始二进制数据来确定这一点。
我的尝试:
#define _CRT_SECURE_NO_WARNINGS //preprocessor requirement
#include <stdio.h> //library for I/O functions
#include <stdlib.h> //library for general functions
#define magic_size 10 //macro for PPM character found within header
typedef struct {
int t_r, t_g, t_b; //Struct to hold RGB pixel data
} Pixel;
int main()
{
char magic_number[magic_size]; //variable for PPM format
int width = 0, height = 0, max_col = 0; //imagine dimensions
int c = 4; //mosaic parameter
/* INPUT FILE HANDLING */
FILE *inputFile;
inputFile = fopen("Sheffield512x512.ppm", "r");
//input file error handling
if (inputFile == NULL)
{
printf(stderr, "ERROR: file cannot be opened");
getchar(); //prevent cmd premature closure
exit(1); //exit program cleanly
}
/* OUTPUT FILE HANDLING */
FILE *outputFile;
outputFile = fopen("mosaic.ppm", "w");
//output file error handling
if (outputFile == NULL)
{
printf(stderr, "ERROR: cannot write to file");
getchar(); //prevent cmd premature closure
exit(1); //exit program cleanly
}
// Scan the header (these variables are used later on)
fscanf(inputFile, "%s\n%d\n%d\n%d", &magic_number, &width, &height, &max_col);
// Error handling. Program only supports binary files (i.e. of P6 format)
if (magic_number[1] != '6')
{
printf("Only Binary images supported!\n");
getchar(); //prevent cmd premature closure
return;
}
// Raw 1 dimensional store of pixel data
Pixel *data = malloc(width*height * sizeof(Pixel));
//2D index to access pixel data
Pixel **pixels = malloc(height * sizeof(Pixel*));
// Read the binary file data
size_t r = fread(data, width*height, sizeof(unsigned char), inputFile);
// Build a 1-dimensional index for the binary data
for (unsigned int i = 0; i < height; ++i)
{
pixels[i] = data + (i * width);
}
// Close the input file
fclose(inputFile);
/* BEGIN PIXELATION PROCESS */
// Print the OUTPUT file header
fprintf(outputFile, "%s\n%d\n%d\n%d", magic_number, width, height, max_col);
//loop condition variables
int cw_x = ceil((double)(width / (float)c));
int cw_y = ceil((double)(height / (float)c));
//iterate through 2d array in cells of size c
for (int c_x = 0; c_x < cw_x; c_x += 1)
{
for (int c_y = 0; c_y < cw_y; c_y += 1)
{
//iterate within the cells
for (int _x = 0; _x < c; _x++)
{
int x = c_x * c + _x;
//bounds checking
if (x < width)
{
for (int _y = 0; _y < c; _y++)
{
int y = c_y * c + _y;
//bounds checking
if (y < height)
{
//write data to the output FILE stream
fwrite(data, width*height, sizeof(unsigned char), outputFile);
}
}
}
}
}
}
//close the output file
fclose(outputFile);
return 0;
}