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

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

Войти
Регистрация
Восстановить пароль
 
VanyakaCompany
3 / 2 / 2
Регистрация: 15.07.2014
Сообщений: 75
#1

Различия между scanf("%s") и gets() - C++

01.06.2016, 11:13. Просмотров 739. Ответов 6

На сколько я знаю, scanf("%s") считывает до первого пробела, а gets() до первого переноса строки.

Недавно заметил такую вещь: gets(), в отличии от scanf("%s"), начинает записывать сразу в 1 строку, при этом нагло пропуская 0.
Я, конечно, нашёл решение - начинать цикл с "-1" ,
но, всё же, интересно узнать: с чем это связано?

Вот код, в котором сравнивается работа двух этих функций:
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
#include "stdafx.h"
#include <iostream>
#include <stdio.h>
#include <cstring>
#include <string.h>
#include <cstdlib>
 
using namespace std;
 
char text1[20][81];
char text2[20][81];
int M;
 
int _tmain(int argc, _TCHAR* argv[])
{
 
    setlocale(0, "");
 
    printf("Введите кол-во строк:\n");
    scanf("%d", &M);
 
    printf("\nВведите текст1:\n");
    for(int i = 0; i < M; i++)
    {
        scanf("%s", text1[i]);
    }
 
    printf("\nВведите текст2:\n");
    for(int i = 0; i < M; i++)
    {
        gets(text2[i]);
    }
 
    printf("\nВаш текст1:\n");
    for(int i = 0; i <  M; i++)
    {
        printf("_ %s\n", text1[i]);
    }
 
    printf("\nВаш текст2:\n");
    for(int i = 0; i <  M; i++)
    {
        printf("_ %s\n", text2[i]);
    }
 
    system("pause");
    return 0;
}
Лучшие ответы (1)
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
01.06.2016, 11:13     Различия между scanf("%s") и gets()
Посмотрите здесь:
C++ Различия между "пробелы" и "табуляция"
C++ scanf (какие различия между %f %g %e)
C++ "cin","cout" на "printf","scanf"
C++ Как сделать так, чтобы scanf в переменную double мог считывать с клавиатуры не только "0,01", но и "0.01"
Расставить между числами знаки "+" и "-" так, чтобы значение выражение стало равно S C++
Кто нибудь делал прогу "Эмулятор клавиш с промежутком времени между "нажатиями""? C++
Создание объекта класса сразу после его описания (между "}" и ";") C++
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Kotik_leapold
73 / 68 / 48
Регистрация: 18.12.2015
Сообщений: 369
01.06.2016, 11:31     Различия между scanf("%s") и gets() #2
Привет, основное отличие:
scanf () заканчивается принимать ввод при встрече с пробелами, символом новой строки или EOF.

gets() рассматривает пробелы как часть входной строки и заканчивается ввод при встрече с новой строки или EOF.

Однако, чтобы избежать ошибок переполнения буфера и избежать угроз безопасности, его безопаснее использовать fgets ()
Humpty
14 / 14 / 5
Регистрация: 10.03.2016
Сообщений: 35
01.06.2016, 11:32     Различия между scanf("%s") и gets() #3
scanf не читает пробельные символы в конце строки. Первое, что видит gets -- '\n', и функция возвращает пустую строку.
VanyakaCompany
3 / 2 / 2
Регистрация: 15.07.2014
Сообщений: 75
01.06.2016, 11:38  [ТС]     Различия между scanf("%s") и gets() #4
А откуда он считывает этот '\n' , ведь я только начинаю вводить строки?

И не приведёт ли к ошибке, если я буду начинать ввод через gets() с (-1) строки?
Humpty
14 / 14 / 5
Регистрация: 10.03.2016
Сообщений: 35
01.06.2016, 11:46     Различия между scanf("%s") и gets() #5
Сообщение было отмечено автором темы, экспертом или модератором как ответ
Это для вас только начинается ввод новой строки, а для компьютера есть два разных потока ввода и вывода. В консоли они перемешаны, но в модели раздельны. Когда вы закончили вводит часть со scanf, в конце повис символ \n. Его gets и читает.

В целом лучше не мешать различные типы ввода/вывода. А в отношении gets использовать fgets. Он безопаснее. Можно хотя бы указать максимальный размер буффера.

Про ошибку не знаю. Вы обращаетесь к text2[-1]. Valgrind на такое, наверное, будет ругаться. А упадет ли программа, я не знаю. Скорее всего здесь будет unpredicted behavior.
VanyakaCompany
3 / 2 / 2
Регистрация: 15.07.2014
Сообщений: 75
01.06.2016, 11:53  [ТС]     Различия между scanf("%s") и gets() #6
Большое спасибо, всё понятно объяснил.

А решение нашёл:
C++
1
2
3
4
5
6
7
gets(text[0]);
 
    printf("\nВведите текст:\n");
    for(int i = 0; i < M; i++)
    {
        gets(text[i]);
    }
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
01.06.2016, 12:04     Различия между scanf("%s") и gets()
Еще ссылки по теме:
C++ Сколько существует способов расставить между цифр знаки "+" и "-"
C++ Почему в scanf("%s",ss) имя ss пишется без амперсанда?
Как с scanf сделать "защиту от дурака"? C++
Как "русификацировать" функцию Scanf() C++
Scanf "chSdSSdbQ" - что такое Q C++

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

Или воспользуйтесь поиском по форуму:
Humpty
14 / 14 / 5
Регистрация: 10.03.2016
Сообщений: 35
01.06.2016, 12:04     Различия между scanf("%s") и gets() #7
Ага, так лучше. Только еще бывают такие неприятные вещи, что в некоторых системах в конце строки стоит \r\n. gets на них может затупить. Если это критично, я бы scanf немного изменил:
C++
1
2
char dummy[256];
scanf("%s%[\n\r\t ]", text1[i], dummy);
%[\n\r\t ] как раз прочитает пробельные символы в буффер dummy.
Yandex
Объявления
01.06.2016, 12:04     Различия между scanf("%s") и gets()
Ответ Создать тему
Опции темы

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