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

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

Войти
Регистрация
Восстановить пароль
 
Infinity20
0 / 0 / 0
Регистрация: 31.01.2013
Сообщений: 25
#1

Выводятся две строки подряд при выводе лишь одной - C++

20.08.2014, 11:23. Просмотров 365. Ответов 7
Метки нет (Все метки)

Баловался со строками. Возник вопрос.
Есть код:
C++
1
2
3
4
5
6
7
8
9
...
char str1[] = "", str2[] = "";
 
cin >> str1;
cout << str1 << endl;
cin >> str2;
cout << str2 << endl;
cout << str1 << endl;
...
Допустим, мы введём в первую строку это: abcde;
Выведет это: abcde.
Вводим данные в str2,например, klmn.
Далее, выведет это:
klmn//cout << str2 << endl;
abcdklmn//cout << str1 << endl;

Почему так работает?cin.clear(); после каждого ввода не помогает.
Ещё заметил, если выделить память динамически, всё работает как надо:

C++
1
2
3
4
5
6
7
8
char* str1 = new char[];
char* str2 = new char[];
 
cin >> str1;
cout << str1 << endl;
cin >> str2;
cout << str2 << endl;
cout << str1 << endl;
В конце выведет это:
abcde // str1
klmn // str2
abcde // str1
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
20.08.2014, 11:23
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Выводятся две строки подряд при выводе лишь одной (C++):

При выводе массива выводятся лишние непонятные символы - C++
Имеется массив vord из 10 символов. Ввожу несколько символов(например 4 символа) в этот массив. Определяю их количество - count. Создаю...

При выводе динамической матрицы выводятся непонятные цифры - C++
данная программа считает суммы строк двумерного массива. но при выводе массива, который хранит их выводятся непонятные цифры( как я...

Почему при записи в файл не выводятся четные строки? - C++
#include &lt;iostream&gt; #include &lt;cctype&gt; #include &lt;fstream&gt; #include &lt;cstdlib&gt; #include &lt;cstring&gt; using namespace std; int...

Ошибка при выводе строки - C++
Подскажите в чем дело?? #include &lt;iostream&gt; #include &lt;conio.h&gt; #include &lt;string.h&gt; using namespace std; class...

Ошибка при выводе строки - C++
С указателями работаю не в первый раз, но почему-то возникает ошибка во время работы. int main() { char temp = &quot;hello_world&quot;; ...

Пропадают строки при выводе в консоль - C++
куда-то пропадают строки. например: for(int q=0;q&lt;10;q++){ cout&lt;&lt;q&lt;&lt;&quot;-q&quot;; for(int w=0;w&lt;10;w++){cout&lt;&lt;w&lt;&lt;&quot;-w&quot;; ...

7
SatanaXIII
Супер-модератор
Эксперт С++
5616 / 2651 / 246
Регистрация: 01.11.2011
Сообщений: 6,534
Завершенные тесты: 1
20.08.2014, 11:37 #2
Цитата Сообщение от Infinity20 Посмотреть сообщение
Почему так работает?
Потому что в конце первого массива не ставится нуль-терминатор.

Добавлено через 2 минуты
Строковые литералы от Evg
0
tehnar5
31 / 31 / 12
Регистрация: 03.05.2011
Сообщений: 84
20.08.2014, 11:38 #3
Представим память в виде массива. По всей видимости, происходит нечто следующее:
Сначала вы берете и помещаете в память две пустые строки. Это значит, что у вас есть два указателя - str1 и str2, которым нужно выделить только одну ячейку памяти(чтобы записать в нее 0 - символ конца строки). Эти блоки памяти могут быть расположены как угодно и где угодно, например, они могут быть расположены близко друг к другу. В Вашем случае они расположены как-то так:
....x...y...., где х - начало первой строки, а у - начало второй строки. Соответственно, после считывания первой строки память выглядит так: .....abcd0...., первая строка начинается с а, вторая с 0. Вывод будет работать корректно, так как он выводит по одному символу, пока не встретится 0. После считывания второй строки память выглядит так: ..abcdklmn0.., вторая строка так же выведется корректно, но теперь, технически, первая строка стала длиннее, так как ближайший справа 0 стал дальше.

Иными словами, если Вы не выделили память/выделили ее мало, то программа вполне может залезть в чужую память. В лучшем случае, это будет память другого процесса и тогда она вылетит с ошибкой, в худшем - это будет какая-то другая переменная. Как мне кажется, второй вариант тоже не правилен, так как надо указывать размер выделяемой памяти, но, по всей видимости, Вам повезло и память выделилась по-другому, подальше друг от друга. Самый правильный способ это new char[maxlength + 1], +1 нужен для того, чтобы хранить 0 - символ конца строки
0
Voivoid
675 / 278 / 12
Регистрация: 31.03.2013
Сообщений: 1,339
20.08.2014, 12:11 #4
Цитата Сообщение от tehnar5 Посмотреть сообщение
это будет память другого процесса
Только если у тебя windows 95

Цитата Сообщение от tehnar5 Посмотреть сообщение
Самый правильный способ это new char[maxlength + 1]
Самый правильный способ это std::string
0
tehnar5
31 / 31 / 12
Регистрация: 03.05.2011
Сообщений: 84
20.08.2014, 12:30 #5
Цитата Сообщение от Voivoid Посмотреть сообщение
Только если у тебя windows 95
Да ладно, ведь если обратиться к str[10000000], то уж наверняка мы выйдем из памяти данного процесса. Другое дело, что система нам не позволит ничего сделать, но, по сути, выход произошел
0
SatanaXIII
Супер-модератор
Эксперт С++
5616 / 2651 / 246
Регистрация: 01.11.2011
Сообщений: 6,534
Завершенные тесты: 1
20.08.2014, 12:48 #6
tehnar5, услышит ли кто звук падающего в лесу дерева, если в лесу никого нет?
0
Voivoid
675 / 278 / 12
Регистрация: 31.03.2013
Сообщений: 1,339
20.08.2014, 12:54 #7
Цитата Сообщение от tehnar5 Посмотреть сообщение
Да ладно, ведь если обратиться к str[10000000], то уж наверняка мы выйдем из памяти данного процесса
Ну, это конечно все зависит от ОС, но если ты про windows, то думаю самое время тебе почитать про механизм виртуальной памяти
0
Alex5
1075 / 739 / 115
Регистрация: 12.04.2010
Сообщений: 1,892
20.08.2014, 15:28 #8
Цитата Сообщение от Infinity20 Посмотреть сообщение
Далее, выведет это:
abcdklmn //cout << str1 << endl;
Как строки расположены в памяти - см. рис.
Кликните здесь для просмотра всего текста
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
    char str1[] = "", str2[] = "";
    cout << "address of str1  " << (void*)&str1[0] << endl;
    cout << "address of str2  " << (void*)&str2[0] << endl;
 
    cin >> str1;
    for( char* p  = &str1[0];  p!= &str1[0]+10 ;  ++p )
    {
        cout << "  " << *p;
    }
    cout << endl;
    //cout << str1 << endl;
    
    cin >> str2;
    for( char* p  = &str1[0];  p!= &str1[0]+10 ;  ++p )
    {
        cout << "  " << *p;
    }
    cout << endl;

Цитата Сообщение от Infinity20 Посмотреть сообщение
выведет abcdklmn
В Вашем случае, Infinity20, видимо, адреса str1 и str2 отличаются на 4 байта. (Подробнее, см. сообщение #3 от tehnar5 ).
0
Миниатюры
Выводятся две строки подряд при выводе лишь одной  
20.08.2014, 15:28
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
20.08.2014, 15:28
Привет! Вот еще темы с ответами:

При выводе пропускает пустые строки - C++
{ FILE *f1; char c, txt; int n=0; cout&lt;&lt;&quot;Vvedite imia fila &quot;; gets(c); f1=fopen(c,&quot;r&quot;); fscanf (f1, &quot;%d&quot;,...

Ограничение ширины строки при выводе в консоль - C++
Класс выводит строку в консоли. Как сделать, чтобы ширина выводимых строк была равна characterWidth = 40, т.е. после 40 символов был...

В файл несколько раз подряд выводятся одинаковые таблицы - C++
Столкнулся с такой проблемой: у меня есть функция ввода данных из файла. Далее эти данные выводятся в консоль и записываются в другой файл....

При попытке вывода строки вместо введенных вручную символов выводятся рандомные - C++
Вся проблема высветлена на скрине. Объясните, пожалуйста, в чем гвоздь. k=300; f=0; dlina=stroka.length(); ...


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

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

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