Форум программистов, компьютерный форум CyberForum.ru

[C++] 24-битные BMP - C++

Восстановить пароль Регистрация
 
 
Рейтинг: Рейтинг темы: голосов - 14, средняя оценка - 4.64
Sahon
10 / 10 / 1
Регистрация: 09.04.2010
Сообщений: 141
17.01.2012, 23:21     [C++] 24-битные BMP #1
Программа не хочет кодировать 24-битные BMP. Под кодированием я понимаю разбитие изображение на квадраты по 10 пикселей (в программе - по 10 байтов, но это не особо существенно потому, что само изображение выходное открываться не хочет), при котором сначала квадраты перемешиваются по вертикали, а потом и по горизонтали. Пример,
1 6 11 16
2 7 12 17
3 8 13 18 =>
4 9 14 19
5 10 15 20

5 6 15 16
4 7 14 17
3 8 13 18 =>
2 9 12 19
1 10 11 20

16 15 6 5
4 7 14 17
18 13 8 3
19 12 9 2
20 11 10 1

P.S. 8-битные битмапы шифруются на ура.

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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
#include <iostream> 
#include <fstream> 
#include <cstdio>
#include <windows.h>
#include <cstring>
 
using namespace std; 
 
int main(int argc, char *argv[]) 
{ 
    setlocale(LC_ALL, "Russian");
    
    if(argc!=2) { //Если более двух аргументов командной строки
        cout << "Используй: next <имя-файла>, будь мужиком!\n"; 
    } 
 
    fstream in(argv[1], ios::in | ios::out | ios::binary); 
    if(!in) { 
        cout << "Невозможно открыть файл.\n";
    } 
  
    BITMAPFILEHEADER bmpFileHeader; //Создание bmpFileHeader
    BITMAPINFOHEADER bmpInfoHeader; //Создание bmpInfoHeader
    
    in.read( (char*)&bmpFileHeader, sizeof( bmpFileHeader ) ); //Считывание данных 
    in.read( (char*)&bmpInfoHeader, sizeof( bmpInfoHeader ) ); //в структуры
    
    const int Height = bmpInfoHeader.biHeight, //Создание констант
              Width = bmpInfoHeader.biWidth,
              BpP = bmpInfoHeader.biBitCount,
              ImageSize = bmpFileHeader.bfSize;
    
    const int RowSize = BpP / 8 * Width ;
    int padding; //Паддинг - ненужное пространсво в массиве пикселей
    
    if (BpP == 8)
        padding = (3 * Width) % 4;
    else if (BpP == 16)
        padding = (2 * Width) % 4;
    else if (BpP == 24)
        padding = Width % 4;
        
    char pixels_orig[Height][RowSize]; //Массив пикселей
    
    in.seekg(bmpFileHeader.bfOffBits, ios::beg); //Перескакиваем в начало массива пикселей
    
    for (int i = 0; i < Height; i++) {
        in.read(pixels_orig[i], (RowSize * (BpP / 8))); //Заполнение одного рядка
        in.seekg(padding, ios::cur); //Проскакиваем паддинг
    }
    
    cout << "Данные " << argv[1] << ":\n"
                 << "\tРазмер файла: " << ImageSize << " байтов = " << (double)ImageSize / 1024 << " килобайтов;\n"
                 << "\tШирина изображения: " << Width << " пикселей;\n"
                 << "\tВысота изображения: " << Height << " пикселей;\n"
                 << "\tБитность: " << BpP << " битов на пиксель.\n";
 
    char temp;
    
    char pixels_crypted[Height][RowSize];
    
    const int code = 10;
    
    for (int i = 0; i < RowSize; i++)
        for (int j = 0; j < Height; j++)
            pixels_crypted[j][i] = pixels_orig[j][i];
    
    //////////////////////  
    //////////////////////      
    ////Новый алгоритм////
    //////////////////////
    //////////////////////
    
    char pattern_1[10][10],
         pattern_2[10][10];
    
    for (int i = 0; i < (Height - Height % 10) / 2 ; i += 10)
        for (int j = 0; j < RowSize - RowSize % 10; j += 20) {
            for (int i_mod_1 = 0; i_mod_1 < 10; i_mod_1++)
                for (int j_mod_1 = 0; j_mod_1 < 10; j_mod_1++) {
                    pattern_1[i_mod_1][j_mod_1] = pixels_crypted[i + i_mod_1][j + j_mod_1];
                    pattern_2[i_mod_1][j_mod_1] = pixels_crypted[Height - Height % 10 - 10 - i + i_mod_1][j + j_mod_1];
                }
                
            for (int i_mod_2 = 0; i_mod_2 < 10; i_mod_2++)  
                for (int j_mod_2 = 0; j_mod_2 < 10; j_mod_2++) {
                    pixels_crypted[i + i_mod_2][j + j_mod_2] = pattern_2[i_mod_2][j_mod_2];
                    pixels_crypted[Height - Height % 10 - 10 - i + i_mod_2][j + j_mod_2] = pattern_1[i_mod_2][j_mod_2];
                }
        }   
    
    for (int i = 0; i < Height - Height % 10 ; i += 20)
        for (int j = 0; j < (RowSize - RowSize % 10) / 2; j += 10) {
            for (int i_mod_1 = 0; i_mod_1 < 10; i_mod_1++)
                for (int j_mod_1 = 0; j_mod_1 < 10; j_mod_1++) {
                    pattern_1[i_mod_1][j_mod_1] = pixels_crypted[i + i_mod_1][j + j_mod_1];
                    pattern_2[i_mod_1][j_mod_1] = pixels_crypted[i + i_mod_1][RowSize - RowSize % 10 - 10 - j + j_mod_1];
                }
                
            for (int i_mod_2 = 0; i_mod_2 < 10; i_mod_2++)  
                for (int j_mod_2 = 0; j_mod_2 < 10; j_mod_2++) {
                    pixels_crypted[i + i_mod_2][j + j_mod_2] = pattern_2[i_mod_2][j_mod_2];
                    pixels_crypted[i + i_mod_2][RowSize - RowSize % 10 - 10 - j + j_mod_2] = pattern_1[i_mod_2][j_mod_2];
                }
        }   
        
    //Создаем файл out.bmp
    fstream out_create("out.bmp", ios::out); 
    out_create.close();
    fstream out("out.bmp", ios::in | ios::out | ios::binary); 
    
    //Начинается запись в файл out.bmp
     
    in.seekg(0, ios::beg);
    
    char ch;
    for (int i = 1; i <= bmpFileHeader.bfOffBits; i++) {
        in.read(&ch, 1);
        out.write(&ch, 1);
    }
    
    in.close();
    
    for (int i = 0; i < Height; i++) {
        out.write(pixels_crypted[i], RowSize); //Заполнение одного рядка
        for (int j = 1; j <= padding; j++)
            out.put(0); //Заполняем паддинг
    }
    
    out.close();
    
    return 0; 
}
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
retmas
Жарю без масла
803 / 685 / 143
Регистрация: 13.01.2012
Сообщений: 1,580
18.01.2012, 01:58     [C++] 24-битные BMP #21
Цитата Сообщение от easybudda Посмотреть сообщение
Ну так функции-то указатели на char принимают
вот именно, то есть в указанную область памяти будут записываться characters

Добавлено через 4 минуты

Не по теме:

Цитата Сообщение от easybudda Посмотреть сообщение
Да и _завтра_ не должно бы
говорят, что все новое - хорошо забытое старое.
а ведь были (а может и сейчас есть?) архитектуры, где тип char состоял из 2х байт.
а вы знаете, что были (а может и сейчас есть?) архитектуры, где байт состоял не из 8 бит, а 9
а вы знаете...

После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9372 / 5422 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
18.01.2012, 02:26     [C++] 24-битные BMP #22
Цитата Сообщение от retmas Посмотреть сообщение
а вы знаете, что были (а может и сейчас есть?) архитектуры, где байт состоял не из 8 бит, а 9
Знаю - из 10 были... когда-то на заре человечества... А ещё (тоже по историческим причинам) функция gethostbyaddr() первым параметром const char* принимает, а передаётся в нём приведённый struct in_addr*, которая (структура) в свою очередь (и снова по историческим причинам) содержит единственное поле типа in_addr_t. Да мало ли ещё чудес на свете, к проблеме-то (напомню - чтение содержимого bmp-файла) это каким боком относится?
retmas
Жарю без масла
803 / 685 / 143
Регистрация: 13.01.2012
Сообщений: 1,580
18.01.2012, 02:54     [C++] 24-битные BMP #23
к тому, что
Цитата Сообщение от easybudda Посмотреть сообщение
sizeof(char) - единица
- непереносимо в общем случае

Добавлено через 9 минут
а если вернуться к 1му посту, то увидим там следущее:
C++
1
2
3
4
5
        BITMAPFILEHEADER bmpFileHeader; //Создание bmpFileHeader
        BITMAPINFOHEADER bmpInfoHeader; //Создание bmpInfoHeader
        
        in.read( (char*)&bmpFileHeader, sizeof( bmpFileHeader ) ); //Считывание данных 
        in.read( (char*)&bmpInfoHeader, sizeof( bmpInfoHeader ) ); //в структуры
скажите чему равен sizeof(BITMAPFILEHEADER) или sizeof(BITMAPINFOHEADER)
вот вам из msdn
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
typedef struct tagBITMAPINFOHEADER { 
  DWORD biSize; 
  LONG biWidth; 
  LONG biHeight; 
  WORD biPlanes; 
  WORD biBitCount 
  DWORD biCompression; 
  DWORD biSizeImage; 
  LONG biXPelsPerMeter; 
  LONG biYPelsPerMeter; 
  DWORD biClrUsed; 
  DWORD biClrImportant; 
} BITMAPINFOHEADER;
C++
1
2
3
4
5
6
7
typedef struct tagBITMAPFILEHEADER {
  WORD bfType; 
  DWORD bfSize; 
  WORD bfReserved1; 
  WORD bfReserved2; 
  DWORD bfOffBits; 
} BITMAPFILEHEADER
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9372 / 5422 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
18.01.2012, 02:55     [C++] 24-битные BMP #24
Цитата Сообщение от retmas Посмотреть сообщение
- непереносимо в общем случае
Так тут-то случай не общий, а вполне конкретный. И размер char на машине ТС с вероятностью 99.(9) равен одному байту. Так и за чем воду мутить?
Я думаю - тут проблема совсем не в чтении данных из файла (раз уж заголовки по словам ТС читаются нормально), а в самом формате 24-битного bmp...
retmas
Жарю без масла
803 / 685 / 143
Регистрация: 13.01.2012
Сообщений: 1,580
18.01.2012, 02:58     [C++] 24-битные BMP #25
я остановлюсь в обсуждении этой темы, так и до китая дойти можно
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9372 / 5422 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
18.01.2012, 03:02     [C++] 24-битные BMP #26
retmas, тогда последний вопрос, и тот скорее риторический - при наличии включённого windows.h о какой переносимости вообще речь?
retmas
Жарю без масла
803 / 685 / 143
Регистрация: 13.01.2012
Сообщений: 1,580
18.01.2012, 17:14     [C++] 24-битные BMP #27
easybudda, хоть я и не собирался возвращаться к этой теме, но всплыла тема и я не удержался. вы хотели пример? думаю он там
п.с. хотя я не уверен, что вы именно это хотели - не разбирался в творчестве
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9372 / 5422 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
18.01.2012, 19:41     [C++] 24-битные BMP #28
Цитата Сообщение от retmas Посмотреть сообщение
вы хотели пример?
Ну там-то проблему Вы и сами определили... Нет, я понял (из вашей же ссылки в 19 посте этой темы), чем не кошерно так читать двоичные данные из файлов, но тут всё-таки другой случай...

Sahon, а не вариант сторонние библиотеки использовать? Вот несколько:
http://easybmp.sourceforge.net/
http://cimg.sourceforge.net/
http://freeimage.sourceforge.net/
Вроде все не сложные...
Sahon
10 / 10 / 1
Регистрация: 09.04.2010
Сообщений: 141
18.01.2012, 22:13  [ТС]     [C++] 24-битные BMP #29
Цитата Сообщение от easybudda Посмотреть сообщение
Ну там-то проблему Вы и сами определили... Нет, я понял (из вашей же ссылки в 19 посте этой темы), чем не кошерно так читать двоичные данные из файлов, но тут всё-таки другой случай...

Sahon, а не вариант сторонние библиотеки использовать? Вот несколько:
http://easybmp.sourceforge.net/
http://cimg.sourceforge.net/
http://freeimage.sourceforge.net/
Вроде все не сложные...
Спасибо, попробуем эти библиотеки заюзать.

Не по теме:

Я просто создавал эту программу с помощью средств Windows и своих собственных только для того, чтобы разобраться во вводе/выводе файлов на примере изображений и для того, чтобы этим самым опыта набратся.

DU
1477 / 1053 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 22:41     [C++] 24-битные BMP #30
Небольшое уточнение:
Сообщение от easybudda
sizeof(char) - единица
Не факт... Да и character и тип char не одно и тоже. Хотя на абсолютном большинстве систем _сегодня_ проблем возникнуть не должно.
Вот это как раз таки факт. В страуструпе написано:
Размеры объектов в с++ выражаются в еденицах размера char. Таким образом, размер char по определению равен 1.
Правда стоит заметить, что из этого не следует, что char в памяти занимает один байт.
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
19.01.2012, 01:00     [C++] 24-битные BMP
Еще ссылки по теме:

16-битные числа C++
Битные маски C++
64 битные типы данных в devC++ C++

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

Или воспользуйтесь поиском по форуму:
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9372 / 5422 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
19.01.2012, 01:00     [C++] 24-битные BMP #31
К&Р глава вторая стих второй:
char - единичный байт, который может содержать один символ из допустимого символьного набора
Yandex
Объявления
19.01.2012, 01:00     [C++] 24-битные BMP
Ответ Создать тему
Опции темы

Текущее время: 20:36. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru