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

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 32, средняя оценка - 4.69
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
#1

Удалить из каждого байта строки первый (старший) бит. - C++

02.11.2011, 22:16. Просмотров 4457. Ответов 19
Метки нет (Все метки)

Здравствуйте.

Суть задачи в общем-то и отображена в названии задачи.

Есть строка: unsigned char str[100];

Необходимо в каждом бите удалить старший бит. Если делать логический сдвиг влево, то я удалю значение старшего бита, а в младший бит занесу "0", но мне надо будет в младший бит заносить старший бит из 2-го байта строки. И так далее. При необходимости последний байт полученной строки дополняется нулевыми битами.

Не могу сообразить как надо организовать перенос битов между байтами. Помогите с идеями пожалуйста..
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
02.11.2011, 22:16
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Удалить из каждого байта строки первый (старший) бит. (C++):

Закодировать файл. Взять 7 байт. Первый бит каждого байта переместить в 8 байт - C++
Нужно открыть бинарный файл и закодировать следующим образом. Взять 7 байт. Первый бит каждого байта переместить в 8 байт. То есть,...

Для целого числа А выделить старший байт и поставить его на место младшего байта. старший байт при этом обнулить. - C++
Помогите пожалуйста, вообще не могу понять как делать следующее...Еще и в программе CODE BLOCKS препод почему именно эту программу...

Бит 8, младший и старший бит (по книге) - C++
Добрый день. Вот читаю книгу, и не пойму элементарной на первый взгляд вещи. Как понять следующие предложения: 1) Если бит...

Старший бит - C++
Допустим число 4 представляю в двоичной системе счисления получается 0100. В етом двоичном числе старший бит ет 3 бит?

Заполнить 16 бит (2 байта) единицами и нулями - C++
Как правильнее и проще заполнить два байта единицами и нулями (то есть задать каждый бит отдельно)? (мне их потом нужно в COM отправить)

Удалить первый нулевой элемент массива и добавить после каждого чётного элемента особый элемент - C++
Здравствуйте! Имеется такая задача: 1) Сформировать одномерный массив целых чисел, используя датчик случайных чисел. 2) Распечатать...

19
fasked
Эксперт С++
4942 / 2522 / 180
Регистрация: 07.10.2009
Сообщений: 4,311
Записей в блоге: 1
02.11.2011, 22:22 #2
ssementsov, битовую маску наложите, не надо никаких сдвигов.
2
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
02.11.2011, 22:25  [ТС] #3
Думал я про маски, но что-то не подумал какую именно надо..

Если бы 2 символа в строке было, то это как-то интуитивно понятно было бы, а если в строке с десяток символов, то уже не догоняю как цикл организовать, чтобы выполнялось требуемое(
0
fasked
Эксперт С++
4942 / 2522 / 180
Регистрация: 07.10.2009
Сообщений: 4,311
Записей в блоге: 1
02.11.2011, 22:45 #4
ssementsov, если я правильно понял, то битовая маска будет 0x7F.
C
1
2
3
4
5
6
7
8
9
10
11
12
#include <stdio.h>
 
int main() {
   unsigned char buf[] = { 0x80, 0x81, 0x82, 0x83 };
   int i = 0;
   
   for (i = 0; i < sizeof(buf)/sizeof(*buf); ++i) {
      printf("%#02x ", (int)(buf[i] & 0x7F));
   }
   
   return 0;
}
2
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
03.11.2011, 00:01  [ТС] #5
Александр, спасибо, сейчас проверю.

Добавлено через 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
#pragma hdrstop
#pragma argsused
 
#include <iostream>
#include <fstream>
#include <conio.h>
#include <iomanip.h>
 
using namespace std;
        void main(){
        clrscr();
        unsigned char str[100];
        long i=0;
 
        cout<<"\nEnter your line: ";
        gets(str);
 
        ofstream outfile("fdata.txt");
        outfile<<endl<<"ÈñõîäГ*Г*Гї ñòðîêГ*: "<<str<<endl;
        outfile<<"ÈñõîäГ*Г*Гї ñòðîêГ* Гў 16-Г© ñèñòåìå ñ÷èñëåГ*ГЁГї:   ";
        while(str[i]){
                outfile<<setiosflags(ios::showbase)<<setiosflags(ios::uppercase)<<hex<<(int)str[i++]<<" ";
        }
 
 
// Вот сюда надо Ваш код вставить.. Чтобы моя строка str преобразовывалась и выводилась в файл.
// Не получилось приспособить( 
 
 
        outfile<<endl<<endl<<"Ïîëó÷åГ*Г*Г*Гї ñòðîêГ* Гў 16-Г© ñèñòåìå ñ÷èñëåГ*ГЁГї: ";
        i=0;
        while(str[i]){
                outfile<<setiosflags(ios::showbase)<<setiosflags(ios::uppercase)<<hex<<(int)str[i++]<<" ";
        }
        outfile<<endl<<"Ïîëó÷åГ*Г*Г*Гї ñòðîêГ*: "<<str<<endl;
        cout<<"\n     Success\n"<<endl<<"(Press any key)";
        getche();
        return;
}
Добавлено через 1 час 5 минут
Не получается почему-то(
0
taras atavin
3569 / 1752 / 91
Регистрация: 24.11.2009
Сообщений: 27,619
03.11.2011, 10:13 #6
Цитата Сообщение от ssementsov Посмотреть сообщение
Удалить из каждого байта строки первый (старший) бит.
Первый - это не старший, а почти младший. Младший нулевой и от него каждый следующий на один шаг старше.
0
fasked
Эксперт С++
4942 / 2522 / 180
Регистрация: 07.10.2009
Сообщений: 4,311
Записей в блоге: 1
03.11.2011, 10:19 #7
ssementsov, вставить надо такой цикл:
C++
1
2
3
        for (int i = 0; i < 100; ++i) {
           str[i] &= 0x7F;
        }
Цитата Сообщение от ssementsov Посмотреть сообщение
outfile<<endl<<"Полученная строка: "<<str<<endl;
Вот так лучше не делать, потому что символы скорее всего получатся непечатаемые.

Не по теме:

taras atavin, а по делу есть что сказать?

1
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
03.11.2011, 11:11  [ТС] #8
Цитата Сообщение от fasked Посмотреть сообщение
ssementsov, вставить надо такой цикл:
C++
1
2
3
        for (int i = 0; i < 100; ++i) {
           str[i] &= 0x7F;
        }
Вот так лучше не делать, потому что символы скорее всего получатся непечатаемые.
Спасибо очередной раз! Сейчас проверю..

А почему будут непечатаемые? В силу кодировок?
В файле отображается нормально сейчас.. Тогда какую альтернативу можно использовать?

Добавлено через 2 минуты
Цитата Сообщение от taras atavin Посмотреть сообщение
Первый - это не старший, а почти младший. Младший нулевой и от него каждый следующий на один шаг старше.
Если читать слева на право, то первый бит - это 7-й. Для этого и указано, что удалить старший бит, т.е. 7-й именно, а не 1-й.

76543210
0
fasked
Эксперт С++
4942 / 2522 / 180
Регистрация: 07.10.2009
Сообщений: 4,311
Записей в блоге: 1
03.11.2011, 12:30 #9
Цитата Сообщение от ssementsov Посмотреть сообщение
А почему будут непечатаемые? В силу кодировок?
В файле отображается нормально сейчас.. Тогда какую альтернативу можно использовать?
Ну вообще, если попадется байт 0x80 (1000 0000b) и из него удалить старший байт, то строка будет выведена в файл не полностью. Все таки нуль-терминированный символ получится.
Для работы с байтовыми массивами лучше всегда использовать функции неформатного ввода/вывода (read/write).

Я все таки считаю, что это именно байтовый массив, а не строка. Битовые махинации для строк не должны использоваться в силу идеологии
1
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
03.11.2011, 20:51  [ТС] #10
Цитата Сообщение от fasked Посмотреть сообщение
Ну вообще, если попадется байт 0x80 (1000 0000b) и из него удалить старший байт, то строка будет выведена в файл не полностью. Все таки нуль-терминированный символ получится.
Для работы с байтовыми массивами лучше всегда использовать функции неформатного ввода/вывода (read/write).

Я все таки считаю, что это именно байтовый массив, а не строка. Битовые махинации для строк не должны использоваться в силу идеологии
Вы абсолютно правы. Суть как раз не в том, что там строка, а в том, что именно байтовый массив. Предмет по которому пишу лабораторную: Теория информации и Кодирования. Это 0-я лабораторная, как я понимаю для обучения работы с битами. Дальше идут лабораторные с кодирование/декодированием и шифрование/дешифрование.

Александр, вот пояснение к задаче..

Суть в том, что когда идут байты:
10001101|01100110|01011001

Биты выделенные красным надо удалить. Не занулить, а именно удалить, т.е. произойдёт сдвиг всех битов влево. В полученном результате синим цветом отметил те биты, которые были перенесены из одного байта в другой. Ну, а фиолетовым выделил те нули, которые надо дописать в конец байта последнего. Если в результате таких вот удалений мы получим просто пустой байт: 00000000, то его и выводить не надо.

00011011|10011010|11001000

Теперь по-моему полностью изложил суть проблемы..

Добавлено через 1 час 6 минут
Делая сдвиг влево, я могу добиться лишь того, что в каждом байте я затру старший бит, остальные биты сдвину, а в нулевой бит запишу "0".

До:
10010101|11101101|11110000
После:
00101010|11011010|11100000

Но дальше не получается сделать необходимое.. Чтобы в биты не "0" писать, а сдвигать биты из следующих байтов..

Добавлено через 1 час 28 минут
Ни у кого нету идей как это реализовать? Сам бы алгоритм понять..
0
accept
4822 / 3243 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
04.11.2011, 06:31 #11
1) сдвинуть в текущем байте влево
2) сдвинуть в следующем байте вправо
3) сдвинутый влево текущий &= сдвинутый вправо следующий
4) взять несдвинутый следующий и сделать его текущим, к п. 1
общее количество бит хранить в переменной
1
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
04.11.2011, 10:59  [ТС] #12
Цитата Сообщение от accept Посмотреть сообщение
1) сдвинуть в текущем байте влево
2) сдвинуть в следующем байте вправо
3) сдвинутый влево текущий &= сдвинутый вправо следующий
4) взять несдвинутый следующий и сделать его текущим, к п. 1
общее количество бит хранить в переменной
Спасибо, сейчас попробую реализовать. Потом отпишусь)
0
accept
4822 / 3243 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
04.11.2011, 11:13 #13
только не &=, а |=
1
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
04.11.2011, 11:21  [ТС] #14
Цитата Сообщение от accept Посмотреть сообщение
только не &=, а |=
В 3-м пункте.. Ок.
0
ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
06.11.2011, 13:04  [ТС] #15
Алгоритм работает неверно( После 3-го пункта на 1-м шаге мы получаем неверный результат.
Пример строка: 012

Строка в двоичном представлении: 01100000 01100001 01100010

После первого шага получаем: 01111000 01100001 01100010

А должны хотя бы что-то вроде: 11000000 01100001 01100010

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
#pragma hdrstop
#pragma argsused
 
#include <iostream>
#include <fstream>
#include <conio.h>
#include <iomanip.h>
 
using namespace std;
 
void main(){
        clrscr();
        unsigned char str[100];
        long i=0;
 
        cout<<"\nEnter your line: ";
        gets(str);
 
        ofstream outfile("fdata.txt");
        outfile<<endl<<"ÈñõîäГ*Г*Гї ñòðîêГ*: "<<str<<endl;
        outfile<<"ÈñõîäГ*Г*Гї ñòðîêГ* Гў 16-Г© ñèñòåìå ñ÷èñëåГ*ГЁГї:   ";
        while(str[i]){
                outfile<<setiosflags(ios::showbase)<<setiosflags(ios::uppercase)<<hex<<(int)str[i++]<<" ";
        }
//----------------
        for(int i = 0; i < strlen(str); i++){
                for(int j = 7; j >= 0; j--){
                     char c = str[i]>>j;
                     c &= 0x01;
                     cout<<(int)c;
                }
                cout<<" ";
        }
 
        cout<<endl<<endl<<endl;
        int len = strlen(str);
 
        for(int i = 0; i < len-1; i+=2){
                str[i] <<= 1;
                str[i+1] >>= 1;
 
                str[i] |= str[i+1];
 
                for(int m = 0; m < len; m++){
                        for(int j = 7; j >= 0; j--){
                                char c = str[m]>>j;
                                c &= 0x01;
                                cout<<(int)c;
                        }
                        cout<<" ";
                }
                cout<<endl<<endl;
        }
 
        for(int i = 0; i < strlen(str); i++){
                for(int j = 7; j >= 0; j--){
                        char c = str[i]>>j;
                        c &= 0x01;
                        cout<<(int)c;
                }
                cout<<" ";
        }
 
//----------------
        outfile<<endl<<endl<<"Ïîëó÷åГ*Г*Г*Гї ñòðîêГ* Гў 16-Г© ñèñòåìå ñ÷èñëåГ*ГЁГї: ";
        i=0;
        while(str[i]){
                outfile<<setiosflags(ios::showbase)<<setiosflags(ios::uppercase)<<hex<<(int)str[i++]<<" ";
        }
        outfile<<endl<<"Ïîëó÷åГ*Г*Г*Гї ñòðîêГ*: "<<str<<endl;
        cout<<"\n     Success\n"<<endl<<"(Press any key)";
        getche();
        return;
}
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
06.11.2011, 13:04
Привет! Вот еще темы с ответами:

Битовое представление каждого байта - C++
Вывести битовое представление каждого байта в интовой переменной. Прошу объясните как это сделать,можно даже без кода,просто не могу понять...

Удалить повторные вхождения каждого слова из строки - C++
Задана строка, состоящая из слов, разделенных одним или несколькими пробелами. Удалить повторные вхождения каждого слова.

Первый бит числа - C++
Пишу программу для сжатия файлов по алгоритму RLE. Чтобы программа понимала, что читаемое число отвечает за количество символов, я решил...

Переставить биты в обратном порядке для каждого байта - C++
Переставить биты в обратном порядке для каждого байта. Решите пожалуйста


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

Или воспользуйтесь поиском по форуму:
15
Yandex
Объявления
06.11.2011, 13:04
Ответ Создать тему
Опции темы

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