程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> 24-Bit BMP Raster Data Tutorial & Grayscaling

24-Bit BMP Raster Data Tutorial & Grayscaling

編輯:C++入門知識

Raster Data Tutorial (24-Bit)
Author: Bill Green (2002)
HOME  EMAIL

This tutorial assumes the reader knows:
(1) Data is stored left to right and bottom to top in a BMP.
(2) How to develop source code to read BMP header and info header (i.e. width, height & # of colors).
(3) How to develop source code to read raster data


INTRODUCTION
There are a possible 256 colors (2^8) that can be stored in a 8-bit image with 255 being the maximum. Likewise, a 24-bit true color BMP has a possible 16 million colors (2^24). In a 24-bit BMP image, a pixel represents a RGB (Red Green Blue) data value. The pixel's RGB data value shows how much Red, Green and Blue are in that particular pixel. One pixel has 3 8-bit colors in it each having an intensity value between 0-255. So a pixel with a data value of (255, 0, 0) is equivalent to (Red=255, Green=0, and Blue=0), or Red! And the composite of the RGB color values produces that pixels actual color. As another example, we know that red and green make yellow. Therefore we would need all red, all green and no blue. Being 255 is the maximum for each color, you would need an RGB data value of (255, 255, 0) to achieve an accurate representation of yellow.
A 24-Bit BMP file structure is slightly different than an 8-Bit BMP file structure. There is no color table for any BMP with a bits/pixel value > 8. The table below shows how the pixel data is stored from the first byte to the last in a 24-Bit BMP.

TABLE 1: 24-Bit BMP File Structure
Byte # to fseek file pointer Information
0
Signature
2
File size
18
Width (number of columns)
22
Height (number of rows)
28
Bits/pixel
46
Number of colors used
54
Start of raster data in a 24-Bit BMP
 
The raster data starts at byte 54. The size of the raster data is (width x height) � 1 bytes. Therefore, a 100 row by 100 column 24-bit image would have (100 x 100) � 1 = 9,999 bytes of raster data starting at byte 54 and continuing to the end of the BMP.

READING 24-Bit BMP RASTER DATA
TEST24.bmp is a 20 row by 20 column BMP image which we will use to read raster data from. The top left portion of TEST24.bmp is yellow and has an RGB pixel value of (255, 255, 0). The bottom right portion is black with an RGB value of (0, 0, 0). The top right portion is green with an RGB value of (0, 255, 0), and the remainder of TEST24.bmp is white (255, 255, 255).

 

(TEST24.bmp is scaled up here to a 100 by 100 BMP,
so be sure and download the zip file to test out your raster data program.)
TEST24.bmp contains 20 rows and 20 columns, so we know we will have 400 bytes of raster data. We also know the raster data will start at byte #54. Knowing this, let抯 try our first program to read raster data and print it to a text file.

To be compiled with Turbo C
Note: download raster24.zip rather than cutting and pasting from below.

#include (stdio.h)
#include (stdlib.h)
#include (math.h)

long getImageInfo(FILE*, long, int);

typedef struct {int rows; int cols; unsigned char* data;} sImage;

int main(int argc, char* argv[])
{
  FILE    *bmpInput, *rasterOutput;
  sImage   originalImage;
  unsigned char   someChar;
  unsigned char   *pChar;
  long    fileSize;
  int    vectorSize, nColors;
  int    r, c;

  /*--------INITIALIZE POINTER----------*/
  someChar = '0';
  pChar = &someChar;

  if(argc < 2)
  {
    printf("Usage: %s bmpInput.bmp\n", argv[0]);
    exit(0);
  }

  printf("Reading file %s\n", argv[1]);

  /*----DECLARE INPUT AND OUTPUT FILES----*/
  bmpInput = fopen(argv[1], "rb");
  rasterOutput = fopen("data24.html", "w");

  fseek(bmpInput, 0L, SEEK_END);

  /*-----GET BMP INFO-----*/
  originalImage.cols = (int)getImageInfo(bmpInput, 18, 4);
  originalImage.rows = (int)getImageInfo(bmpInput, 22, 4);
  fileSize = getImageInfo(bmpInput, 2, 4);
  nColors = getImageInfo(bmpInput, 46, 4);

  /*----PRINT BMP INFO TO SCREEN-----*/
  printf("Width: %d\n", originalImage.cols);
  printf("Height: %d\n", originalImage.rows);
  printf("File size: %ld\n", fileSize);
  printf("Bits/pixel: %d\n", getImageInfo(bmpInput, 28, 4));
  printf("No. colors: %d\n", nColors);


  /*----FOR 24-BIT BMP, THERE IS NO COLOR TABLE-----*/
  fseek(bmpInput, 54, SEEK_SET);

  /*-----------READ RASTER DATA-----------*/
  for(r=0; r<=originalImage.rows-1; r++)
  {
    for(c=0; c<=originalImage.cols-1; c++)
    {
     
      /*----READ FIRST BYTE TO GET BLUE VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      blueValue = *pChar;

      /*-----READ NEXT BYTE TO GET GREEN VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      greenValue = *pChar;

      /*-----READ NEXT BYTE TO GET RED VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      redValue = *pChar;

      /*---------PRINT TO TEXT FILE---------*/
      fprintf(rasterOutput, "(%d %d) = \tRed \t%d", r, c, redValue);
      fprintf(rasterOutput, "\tGreen \t%d \tBlue \t%d\n", greenValue, blueValue);

    }
  }

  fclose(bmpInput);
  fclose(rasterOutput);

  return 0;
}

/*--------SUBPROGRAMS------------*/

long getImageInfo(FILE* inputFile, long offset, int numberOfChars)
{
  unsigned char   *ptrC;
  long    value=0L;
  int    i;
  unsigned char   dummy;


  dummy = '0';
  ptrC = &dummy;

  fseek(inputFile, offset, SEEK_SET);

  for(i=1; i<=numberOfChars; i++)
  {
    fread(ptrC, sizeof(char), 1, inputFile);
    /* calculate value based on adding bytes */
    value = (long)(value + (*ptrC)*(pow(256, (i-1))));
  }

  return(value);
}

Running your raster data program, you will get an ASCII file called data24.txt with some entries looking like the following:


(0 0) =  Red  255 Green  255  Blue  255
(0 1) =  Red  255 Green  255  Blue  255
(0 2) =  Red  255 Green  255  Blue  255
(0 3) =  Red  255 Green  255  Blue  255
(0 4) =  Red  255 Green  255  Blue  255
(0 5) =  Red  255 Green  255  Blue  255
(0 6) =  Red  255 Green  255  Blue  255
(0 7) =  Red  255 Green  255  Blue  255
(0 8) =  Red  255 Green  255  Blue  255
(0 9) =  Red  255 Green  255  Blue  255
(0 10) =  Red  255 Green  255  Blue  255
(0 11) =  Red  255 Green  255  Blue  255
(0 12) =  Red  0 Green  0  Blue  0
(0 13) =  Red  0 Green  0  Blue  0
(0 14) =  Red  0 Green  0  Blue  0
(0 15) =  Red  0 Green  0  Blue  0
(0 16) =  Red  0 Green  0  Blue  0
(0 17) =  Red  0 Green  0  Blue  0
(0 18) =  Red  0 Green  0  Blue  0
(0 19) =  Red  0 Green  0  Blue  0

  :

(8 0) =  Red  255 Green  255  Blue  0
(8 1) =  Red  255 Green  255  Blue  0
(8 2) =  Red  255 Green  255  Blue  0
(8 3) =  Red  255 Green  255  Blue  0
(8 4) =  Red  255 Green  255  Blue  0
(8 5) =  Red  255 Green  255  Blue  0
(8 6) =  Red  255 Green  255  Blue  0
(8 7) =  Red  255 Green  255  Blue  0
(8 8) =  Red  255 Green  255  Blue  0
(8 9) =  Red  255 Green  255  Blue  0
(8 10) =  Red  255 Green  255  Blue  0
(8 11) =  Red  255 Green  255  Blue  0
(8 12) =  Red  255 Green  255  Blue  255
(8 13) =  Red  255 Green  255  Blue  255
(8 14) =  Red  255 Green  255  Blue  255
(8 15) =  Red  255 Green  255  Blue  255
(8 16) =  Red  255 Green  255  Blue  255
(8 17) =  Red  255 Green  255  Blue  255
(8 18) =  Red  255 Green  255  Blue  255
(8 19) =  Red  255 Green  255  Blue  255

  :

(16 0) =  Red  255 Green  255  Blue  0
(16 1) =  Red  255 Green  255  Blue  0
(16 2) =  Red  255 Green  255  Blue  0
(16 3) =  Red  255 Green  255  Blue  0
(16 4) =  Red  255 Green  255  Blue  0
(16 5) =  Red  255 Green  255  Blue  0
(16 6) =  Red  255 Green  255  Blue  0
(16 7) =  Red  255 Green  255  Blue  0
(16 8) =  Red  255 Green  255  Blue  0
(16 9) =  Red  255 Green  255  Blue  0
(16 10) =  Red  255 Green  255  Blue  0
(16 11) =  Red  255 Green  255  Blue  0
(16 12) =  Red  255 Green  255  Blue  255
(16 13) =  Red  255 Green  255  Blue  255
(16 14) =  Red  0 Green  255  Blue  0
(16 15) =  Red  0 Green  255  Blue  0
(16 16) =  Red  0 Green  255  Blue  0
(16 17) =  Red  0 Green  255  Blue  0
(16 18) =  Red  0 Green  255  Blue  0
(16 19) =  Red  0 Green  255  Blue  0
  :


Notice how entry (16, 0) is (Red = 255, Green = 255, Blue = 0) corresponding to a yellow pixel in TEST24.bmp. And we can see that looking at TEST24.bmp, it matches precisilely. Just remember that in BMPs, the raster data is stored from left to right and bottom to top - so row 16, column 0 is somewhere in the the top left corner!

EXPLANATION
To get preliminary code explanation, please see my raster data tutorial. The difference between this code and my 8-Bit raster code is that instead of 1 byte representing a pixel (8-bit), we now have 3 bytes representing each pixel (24-bit). So we cannot read the value of the pixel intensity and then move on to the next row-column entry. We have to read 3 different bytes! The algorithm below was used to do this:

      /*----READ FIRST BYTE TO GET BLUE VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      blueValue = *pChar;

      /*-----READ NEXT BYTE TO GET GREEN VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      greenValue = *pChar;

      /*-----READ NEXT BYTE TO GET RED VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      redValue = *pChar;

      /*---------PRINT TO TEXT FILE---------*/
      fprintf(rasterOutput, "(%d %d) = \tRed \t%d", r, c, redValue);
      fprintf(rasterOutput, "\tGreen \t%d \tBlue \t%d\n", greenValue, blueValue);   
CONVERTING A 24-BIT BMP TO GRAY SCALE
We want to take the TEST24.bmp and grayscale it, or convert it from a 24-bit to an 8-bit BMP. We'll call our new grayscaled BMP "gray24.bmp". A formula for converting a RGB pixel value to a grayscale value is shown below:
grayValue = 0.299*redValue + 0.587*greenValue + 0.114*blueValue
First, I copied the header and info header from the input BMP to the output BMP. This process is described in my raster data tutorial. They are the same with the exception of the file size, bits/pixel value, and the number of colors. We manipulate them by using the following code

  /*----CHANGE BIT DEPTH FROM 24 TO 8----*/
  fseek(bmpOutput, 28, SEEK_SET);
  *pLong = (unsigned long)(8);
  fwrite(pLong, sizeof(unsigned long), 1, bmpOutput);
The color tables, however, are not identical. There is no color table in a 24-bit BMP (see above), while there is one in a grayscale image. Therefore, instead of copying the color table like we have been, we are actually going to have to create one. I did this by just copying the color table of another grayscale BMP.

createColorTable(grayBmpInput, bmpOutput);
The following portion of code converts a 24-Bit BMP file to grayscale. The differences from raster24.c are in red.

To be compiled with Turbo C
Note: download gray24.zip rather than cutting and pasting from below.
       /*-----READ FIRST BYTE TO GET BLUE VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      blueValue = *pChar;

      /*-----READ NEXT BYTE TO GET GREEN VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      greenValue = *pChar;

      /*-----READ NEXT BYTE TO GET RED VALUE-----*/
      fread(pChar, sizeof(char), 1, bmpInput);
      redValue = *pChar;www.2cto.com

      /*-----USE FORMULA TO CONVERT RGB VALUE TO GRAYSCALE-----*/
      grayValue = (int)(0.299*redValue + 0.587*greenValue + 0.114*blueValue);

      /*-----PRINT TO TEXT FILE-----*/
      fprintf(rasterOutput, "(%d %d) = \tRed \t%d", r, c, redValue);
      fprintf(rasterOutput, "\tGreen \t%d \tBlue \t%d \tGray \t%d\n", greenValue, blueValue, grayValue);

      /*-----WRITE TO NEW BMP FILE------*/
      *pChar = grayValue;
      fseek(bmpOutput, (54 + 4*256), SEEK_SET);
      fwrite(pChar, sizeof(char), 1, bmpOutput);


作者:caiye917015406

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved