Форум программистов, компьютерный форум, киберфорум
Наши страницы

C++

Войти
Регистрация
Восстановить пароль
 
Рейтинг: Рейтинг темы: голосов - 46, средняя оценка - 5.00
solyr
1 / 1 / 0
Регистрация: 14.05.2010
Сообщений: 12
#1

Чтение BMP файла - C++

01.12.2010, 16:52. Просмотров 6247. Ответов 1
Метки нет (Все метки)

Пишу программу в которой необходимо считать заголовок BMP файла. Файл безпалитровый, глубина цвета 24, то есть по 8 бит каждой составляющей RGB. Заголовок считывается не правильно почему то. Структура заголовка у меня объявлена правильно.

Считываю вызовом fread(&BMPHeader,sizeof(TBMPHeader),1,f);

Мне надо сохранить каждый канал изображения (красный, зеленый и синий) как отдельное растровое изображение.

Создаю три файла и записываю в них считанный заголовок, далее для каждого отдельного канала обнуляю две составляющии RGB, что получить одноканальное изображение. У меня все правильно записывается тока размеры файлов почему огромные. За 100 мегабит превышают, хотя исходный файл считывается весит всего 2-3 метра.

Формат заголовка:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
struct TBMPHeader{
    //заголовок файла
    short Type;          //сигнатура BM 0x4D42
    long  Size;          //размер файла в байтах 3*H*W+54
    long  Reserved;      //зарезервировано 0
    long  OffsetBits;    //смещение изображения от начала файла 54
    //информационный заголовок
    long  SizeH;         //размер заголовка 40 информационного заголовка
    long  Width;         //ширина изображения
    long  Height;        //высота изображения   
    short Planes;        //число плоскостей 1
    short BitCount;      //глубина цвета, бит на точку 24
    long  Compression;   //тип еомпрессии(0 - несжатое изображение)
    long  SizeImage;     //разиер изображения, байт 3*W*H
    long  XPelsPerMeter; //горизонтальное разрешение, точка на метр 0
    long  YPelsPerMeter; //вертикальное разрешение, точка на метр 0
    long  ColorUsed;     //число используемых цветов 0
    long  ColorImportant;//число основных цветов 0
};
Привожу исходный код процедуры: Проца пока реализована без выравнивания по щирине. Там кратность 4 не пока не учитывается. Засылаю входной файл сразу кратный 4 ем ширине. После допишу что нужно, но вот ошибка с размером и чет заголок не правильно считывается

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
void ConvertBMPFile(char *fname){
    TBMPHeader BMPHeader;
    FILE *f,*rf,*gf,*bf;
    if ((f = fopen(fname,"rb")) == NULL){
        cout<<"Ошибка открытия файла";
        return;
    }
    fread(&BMPHeader,sizeof(TBMPHeader),1,f);
    //if (Header.BitCount == 24) {
        if ((rf = fopen("red.bmp","wb")) == NULL){
            cout<<"Ошибка открытия файла";
            return;
        }
        if ((gf = fopen("green.bmp","wb")) == NULL){
            cout<<"Ошибка открытия файла";
            return;
        }
        if ((bf = fopen("blue.bmp","wb")) == NULL){
            cout<<"Ошибка открытия файла";
            return;
        }
        fwrite(&BMPHeader,sizeof(TBMPHeader),1,rf);
        fwrite(&BMPHeader,sizeof(TBMPHeader),1,gf);
        fwrite(&BMPHeader,sizeof(TBMPHeader),1,bf);
        
        TBGR *inBGR  = new TBGR[BMPHeader.Width]; //массив указателей на слова
        TBGR *inBGRr = new TBGR[BMPHeader.Width];
        TBGR *inBGRg = new TBGR[BMPHeader.Width];
        TBGR *inBGRb = new TBGR[BMPHeader.Width];
 
        cout<<"Ждем ...";
 
        while(!feof(f)){
            fread(inBGR,3*BMPHeader.Width,1,f);
            for(int i=0;i<BMPHeader.Width-1;i++){
                inBGRr[i] = inBGR[i];
                inBGRg[i] = inBGR[i];
                inBGRb[i] = inBGR[i];
            }   
            for(int i=0;i<BMPHeader.Width-1;i++){
                inBGRr[i].B = 0;
                inBGRr[i].G = 0;
            }   
            for(int i=0;i<BMPHeader.Width-1;i++){
                inBGRg[i].B = 0;
                inBGRg[i].R = 0;
            }   
            for(int i=0;i<BMPHeader.Width-1;i++){
                inBGRb[i].R = 0;
                inBGRb[i].G = 0;
            }   
            fwrite(inBGRr,3*BMPHeader.Width,1,rf);
            fwrite(inBGRg,3*BMPHeader.Width,1,gf);
            fwrite(inBGRb,3*BMPHeader.Width,1,bf);
        }
        fclose(rf);
        fclose(gf);
        fclose(bf);
    //} else {
    //  cout<<"Файл палитровый !!!";
    //}
    fclose(f);
}
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
01.12.2010, 16:52
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Чтение BMP файла (C++):

Чтение заголовка BMP - C++
Есть след код, который должен читать заголовок BMP Но он его читаеть криво (пропускает 2 байта) Ктонить может помочь решить эту...

Загрузка и сохранение BMP файла - C++
Добрый день! Мне нужно загружать, а затем сохранять BMP файл, пока без обработки, просто загрузить в массив и потом из этого массива...

Правильное чтение из файла! - C++
Помогите пожалуйста, уже опробовал все((( Суть задачи такая, есть файл data.txt в нем есть числа выглядит он так: Ниже представлен...

Не работает на С++ чтение с файла в сменную - C++
Всем здрасте. Я делаю курсовую. И вот такое. У меня есть структура: struct Catalog { string PName; string Pnaz; string...

OpenCV. Некорректное чтение бинарного файла - C++
Здравствуйте. При использовании opencv (2.4.6) столкнулся со следующей особенностью (или багом). Имеются бинарные файлы (fits-файлы с...

Быстрое чтение большого файла. Зависает процесс - C++
Всем доброго времени суток. Есть задача прочитать файл(1-20 мб), который состоит из слов разделённых пробелами для того, чтобы переписать...

1
solyr
1 / 1 / 0
Регистрация: 14.05.2010
Сообщений: 12
01.12.2010, 20:12  [ТС] #2
Вроде как с размерами разобрались. Пробла возникла с некратными 4 ем по ширине файлами. Кратные правильно конвертирует,а вот не кратные не правильно. Портит файл. Не могу понять где ошибка в коде конвертации. Помогите найти ошибку!!!

Выкладываю код:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
void ConvertBMPFileToMonoCannal(char *fname){
    TBMPHeader BMPHeader;
    FILE *f,*rf,*gf,*bf;
    if ((f = fopen(fname,"rb")) == NULL){
        cout<<"Ошибка открытия файла";
        return;
    }
    fread(&BMPHeader,sizeof(TBMPHeader),1,f);
    if (BMPHeader.BitCount == 24) {
        
        if ((rf = fopen("red.bmp","wb")) == NULL){
            cout<<"Ошибка открытия файла";
            return;
        }
        if ((gf = fopen("green.bmp","wb")) == NULL){
            cout<<"Ошибка открытия файла";
            return;
        }
        if ((bf = fopen("blue.bmp","wb")) == NULL){
            cout<<"Ошибка открытия файла";
            return;
        }
        fwrite(&BMPHeader,sizeof(TBMPHeader),1,rf);
        fwrite(&BMPHeader,sizeof(TBMPHeader),1,gf);
        fwrite(&BMPHeader,sizeof(TBMPHeader),1,bf);
        
        int w = BMPHeader.Width / 4; 
        if ((BMPHeader.Width % 4) > 0) w++;
        int Width = w*4;
                
        TBGR *inBGR  = new TBGR[Width]; //массив указателей на слова
        TBGR *inBGRr = new TBGR[Width];
        TBGR *inBGRg = new TBGR[Width];
        TBGR *inBGRb = new TBGR[Width];
        
        for (int j = 0; j < BMPHeader.Height; j++){
 
            fread(inBGR,3*Width,1,f);
 
            for(int i=0;i < Width; i++){
                inBGRr[i] = inBGR[i];
                inBGRr[i].B = 0;
                inBGRr[i].G = 0;
 
                inBGRg[i] = inBGR[i];
                inBGRg[i].B = 0;
                inBGRg[i].R = 0;
 
                inBGRb[i] = inBGR[i];
                inBGRb[i].R = 0;
                inBGRb[i].G = 0;
            }   
 
            fwrite(inBGRr,3*Width,1,rf);
            fwrite(inBGRg,3*Width,1,gf);
            fwrite(inBGRb,3*Width,1,bf);
        }
        fclose(rf);
        fclose(gf);
        fclose(bf);
    } else  cout<<"Файл палитровый !!!";
    fclose(f);
}
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
01.12.2010, 20:12
Привет! Вот еще темы с ответами:

Чтение бинарного файла в строку: читается только один байт - C++
Читаю бинарный файл в строку, но почему-то читает только 1 байт. Где здесь может быть ошибка? //ЧТЕНИЕ БИНАРНОГО ФАЙЛА В СТРОКУ FILE *f...

Чтение файла (с кракозябрами) с однобайтовой кодировкой, конвертация в UTF-8 и вывод в другой файл - C++
Добрый день, форумчане. Может у кого есть заготовка или кто поделится советом, необходимо прочитать файл (с кракозябрами) с однобайтовой...

Отразить bmp - C++
Здравствуйте, подскажите, как можно отразить bmp по вертикали или горизонтали? На WinAPI. Желательно код. Спасибо.

Работа с BMP форматом - C++
Всем доброго времени суток Проблема такая: начал разбираться со струтурой BMP формата пока описал только структуру BMP и считал...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
2
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru