Результат использования strtol() в stdlib.h и stdio.h отличается

При попытке разобрать число, слишком большое для длинного, strtol() возвращает 0 вместо LONG_MAX (stdio.h). Если я правильно прочитал спецификацию POSIX, это должно быть LONG_MAX. Между stdio.h и stdlib.h есть разница

#include "stdio.h"
int main(void){
    printf("%ld\n", strtol("99999999999999999999999"));
    return 0;
} # 0
#include "stdio.h"
//#include "stdlib.h"
int main(void){
    char *end[500];
    printf("%ld\n", strtol("99999999999999999999999", end, 10));
    return 0;
} # 9223372036854775807

person Zone233    schedule 06.02.2019    source источник


Ответы (1)


strtol объявляется в заголовке <stdlib.h> как

long strtol( const char *restrict str, char **restrict str_end, int base );
//                       ^^^^^^^^             ^^^^^^^^ since C99

В первом опубликованном фрагменте <stdlib.h> не включено, и функция вызывается с одним единственным аргументом, так что при компиляции с -Wall -Wextra -std=gnu11 gcc перед выводом 0 выдает следующие поясняющие предупреждения:

prog.c: In function 'main':
prog.c:5:21: warning: implicit declaration of function 'strtol' [-Wimplicit-function-declaration]
     printf("%ld\n", strtol("99999999999999999999999"));
                     ^~~~~~
prog.c:5:15: warning: format '%ld' expects argument of type 'long int', but argument 2 has type 'int' [-Wformat=]
     printf("%ld\n", strtol("99999999999999999999999"));
             ~~^     ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
             %d

Это означает, что библиотечная функция вообще не вызывается, а вызывается неявно объявленная функция с тем же именем, возвращается и печатается int со значением 0 (с неправильным спецификатором формата, что само по себе является неопределенным поведением).

Обратите внимание, что тот же код не компилируется с clang, который сообщает следующее:

prog.c:4:21: warning: implicitly declaring library function 'strtol' with type 'long (const char *, char **, int)' [-Wimplicit-function-declaration]
    printf("%ld\n", strtol("99999999999999999999999"));
                    ^
prog.c:4:21: note: include the header <stdlib.h> or explicitly provide a declaration for 'strtol'
prog.c:4:53: error: too few arguments to function call, expected 3, have 1
    printf("%ld\n", strtol("99999999999999999999999"));
                    ~~~~~~                          ^
1 warning and 1 error generated.

Во втором фрагменте strtol вызывается с правильным количеством аргументов, но, как указано (с закомментированным #include), имеет ту же проблему с отсутствующим заголовком. Чтобы получить ожидаемый результат, LONG_MAX, заголовок stdlib.h должен быть включен.

person Bob__    schedule 06.02.2019