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

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

Восстановить пароль Регистрация
 
Рейтинг: Рейтинг темы: голосов - 32, средняя оценка - 4.69
ssementsov
 Аватар для ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
02.11.2011, 22:16     Удалить из каждого байта строки первый (старший) бит. #1
Здравствуйте.

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

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

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

Не могу сообразить как надо организовать перенос битов между байтами. Помогите с идеями пожалуйста..
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
fasked
Эксперт C++
 Аватар для fasked
4924 / 2504 / 180
Регистрация: 07.10.2009
Сообщений: 4,306
Записей в блоге: 1
02.11.2011, 22:22     Удалить из каждого байта строки первый (старший) бит. #2
ssementsov, битовую маску наложите, не надо никаких сдвигов.
ssementsov
 Аватар для ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
02.11.2011, 22:25  [ТС]     Удалить из каждого байта строки первый (старший) бит. #3
Думал я про маски, но что-то не подумал какую именно надо..

Если бы 2 символа в строке было, то это как-то интуитивно понятно было бы, а если в строке с десяток символов, то уже не догоняю как цикл организовать, чтобы выполнялось требуемое(
fasked
Эксперт C++
 Аватар для fasked
4924 / 2504 / 180
Регистрация: 07.10.2009
Сообщений: 4,306
Записей в блоге: 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;
}
ssementsov
 Аватар для 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 минут
Не получается почему-то(
taras atavin
Ушёл с форума.
 Аватар для taras atavin
3569 / 1752 / 91
Регистрация: 24.11.2009
Сообщений: 27,619
03.11.2011, 10:13     Удалить из каждого байта строки первый (старший) бит. #6
Цитата Сообщение от ssementsov Посмотреть сообщение
Удалить из каждого байта строки первый (старший) бит.
Первый - это не старший, а почти младший. Младший нулевой и от него каждый следующий на один шаг старше.
fasked
Эксперт C++
 Аватар для fasked
4924 / 2504 / 180
Регистрация: 07.10.2009
Сообщений: 4,306
Записей в блоге: 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, а по делу есть что сказать?

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

Я все таки считаю, что это именно байтовый массив, а не строка. Битовые махинации для строк не должны использоваться в силу идеологии
ssementsov
 Аватар для 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 минут
Ни у кого нету идей как это реализовать? Сам бы алгоритм понять..
accept
4838 / 3237 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
04.11.2011, 06:31     Удалить из каждого байта строки первый (старший) бит. #11
1) сдвинуть в текущем байте влево
2) сдвинуть в следующем байте вправо
3) сдвинутый влево текущий &= сдвинутый вправо следующий
4) взять несдвинутый следующий и сделать его текущим, к п. 1
общее количество бит хранить в переменной
ssementsov
 Аватар для ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
04.11.2011, 10:59  [ТС]     Удалить из каждого байта строки первый (старший) бит. #12
Цитата Сообщение от accept Посмотреть сообщение
1) сдвинуть в текущем байте влево
2) сдвинуть в следующем байте вправо
3) сдвинутый влево текущий &= сдвинутый вправо следующий
4) взять несдвинутый следующий и сделать его текущим, к п. 1
общее количество бит хранить в переменной
Спасибо, сейчас попробую реализовать. Потом отпишусь)
accept
4838 / 3237 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
04.11.2011, 11:13     Удалить из каждого байта строки первый (старший) бит. #13
только не &=, а |=
ssementsov
 Аватар для ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
04.11.2011, 11:21  [ТС]     Удалить из каждого байта строки первый (старший) бит. #14
Цитата Сообщение от accept Посмотреть сообщение
только не &=, а |=
В 3-м пункте.. Ок.
ssementsov
 Аватар для 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;
}
accept
4838 / 3237 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
06.11.2011, 13:37     Удалить из каждого байта строки первый (старший) бит. #16
Цитата Сообщение от ssementsov
Пример строка: 012
Строка в двоичном представлении: 01100000 01100001 01100010
если строка в десятичной системе, то 00001100
если строка в восьмеричной системе, то 00001010
ssementsov
 Аватар для ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
06.11.2011, 13:57  [ТС]     Удалить из каждого байта строки первый (старший) бит. #17
Цитата Сообщение от accept Посмотреть сообщение
если строка в десятичной системе, то 00001100
если строка в восьмеричной системе, то 00001010
Вы думаете, что 012 - это число. Но это строка из 3-х символов: "0", "1", "2"

Для каждого символа свой код в ASCII-таблице. Например, для "0" - это 01100000. И так далее. Вот именно эту последовательность битов и надо обрабатывать.
accept
4838 / 3237 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
06.11.2011, 14:08     Удалить из каждого байта строки первый (старший) бит. #18
Цитата Сообщение от ssementsov
C++
1
2
                str[i] <<= 1;
                str[i+1] >>= 1;
сдвигать вправо надо не на один бит, а пока не останется один бит
благодаря беззнаковому типу, при правом сдвиге слева будут нули

ещё там нет обработки последнего байта
его тоже надо сдвигать влево
ssementsov
 Аватар для ssementsov
0 / 0 / 0
Регистрация: 02.11.2011
Сообщений: 89
09.11.2011, 20:15  [ТС]     Удалить из каждого байта строки первый (старший) бит. #19
Ребят, вот так получилось у меня..
Вроде работает.
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
#pragma hdrstop
#pragma argsused
#include <iostream>
#include <fstream>
#include <conio.h>
#include <iomanip.h>
#include <math.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++]<<" ";
        }
//-----------------------------------------------------------------------------------------------------------
        long n=strlen(str);
    for(long i = 0; i < n; i++){
        str[i] <<= 1;
    }
    long sdv = 1, j = -1;
 
    for(long i = 0; i < n; i++){
        unsigned char z = str[i+1];
        j = j >= 7 ? 0 : j+1;
        z >>= (7-j);
        int minus = 0;
        for(long p = 0; p <= j; p++){
            minus += pow(2, (double)p);
        }
 
        str[i] &= 255-minus;
        str[i] |= z;
        if(sdv > 7) sdv = 0;
        str[i+1] <<= sdv++;
    }
    cout<<endl;
//-----------------------------------------------------------------------------------------------------------
    n = (n%8 == 1 || n<8) ? n : n-1;
        outfile<<endl<<endl<<"Полученная строка в 16-й системе счисления: ";
 
        for(i=0; i<n; 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 минуту
Может улучшить можно как-либо?
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
09.11.2011, 21:25     Удалить из каждого байта строки первый (старший) бит.
Еще ссылки по теме:

C++ Первый бит числа
C++ Закодировать файл. Взять 7 байт. Первый бит каждого байта переместить в 8 байт
Битовое представление каждого байта C++

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

Или воспользуйтесь поиском по форуму:
Evg
Эксперт С++Автор FAQ
 Аватар для Evg
16824 / 5245 / 320
Регистрация: 30.03.2009
Сообщений: 14,125
Записей в блоге: 26
09.11.2011, 21:25     Удалить из каждого байта строки первый (старший) бит. #20
Когда голова плохо соображает, как такое сделать программно (хотя понимает, что сделать на бумажке), зачастую хорошим выходом будут череж...ые решения. Они неэффективные, некрасивые, зато понятные. Здесь я пользуюсь двумя разными понятиями "байт" и "char" (что есть тоже байт), поскольку они описывают немного разные сущности.

Берём исходный char и распиливаем его на отдельные биты и в результате создаём массив из 8 байт, в каждом элементе которого записан 0 или 1, в соответствии с нужным битом. Работаем кусками по 8 char'ов. Т.е. 8 char'ов превращаем в массив из 64 байтов. Далее из этого массива формируем массив из 56 байт, в котором выкинуты каждый 8-й байт (что соотвествует каждому старшему биту из каждого исходного char'а). Далее эти 56 байт превращаем в 7 char'ов. Таким образом из исходных 8 char'ов мы получили результирующие 7 char'ов, которые отображают последовательность с выкинутыми битами.

Если количество char'ов в исходном коде не кратно 8, то их искусственно дорастить нулями до кратного 8 значения, а потом посчитать, сколько выкинуть из хвоста результирующей последовательности

Поскольку в программе ты будешь оперировать одним и тем же типом для того, что я называл "char" и "байт", то вместо "байт" лучше использовать int. Отожрётся немного побольше памяти, но зато не запутаешься, который char что обозначает
Yandex
Объявления
09.11.2011, 21:25     Удалить из каждого байта строки первый (старший) бит.
Ответ Создать тему
Опции темы

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