asprintf - как получить строковый ввод в C

Я читаю книгу "21 век C" (первое редактирование) и нахожу интересную программу, использующую asprintf для получения строки без использования malloc /size of для длины строки или выделения пространства. Пожалуйста, прочитайте прикрепленное изображение из той же книги, чтобы понять контекст. Следующие программы также из книги. Программа компилируется и НЕ принимает строковый ввод с клавиатуры, а получает следующее сообщение. Вопрос в том, почему программа не принимает строковый ввод с keboard, а показывает длинное (необычное) сообщение об ошибке?

#define _GNU_SOURCE  // stdio.h to include asprintf
#include <stdlib.h>
#include <stdio.h>

void get_strings(char const *in) {
    char *cmd;
    asprintf(&cmd, "strings %s", in);
    if (system(cmd))
        fprintf(stderr, "Something went Wrong  %s.\n", cmd);
    free(cmd);
} 

int main(int argc, char **argv) {
    get_strings(argv[0]);   
    //return 0;
}

При запуске программы вывод:

/lib64/ld-linux-x86-64.so.2
libc.so.6
__stack_chk_fail
asprintf
stderr
system
fprintf
__libc_start_main
free
__gmon_start__
GLIBC_2.4
GLIBC_2.2.5
UH-X
AWAVA
AUATL
[]A\A]A^A_
strings %s
Something  went Wrong  %s.
;*3$"
GCC: (Ubuntu 5.3.1-14ubuntu2) 5.3.1 20160413
crtstuff.c
__JCR_LIST__
deregister_tm_clones
__do_global_dtors_aux
completed.7585
__do_global_dtors_aux_fini_array_entry
frame_dummy
__frame_dummy_init_array_entry
get_strings.c
__FRAME_END__
__JCR_END__
__init_array_end
_DYNAMIC
__init_array_start
__GNU_EH_FRAME_HDR
_GLOBAL_OFFSET_TABLE_
__libc_csu_fini
free@@GLIBC_2.2.5
_ITM_deregisterTMCloneTable
_edata
__stack_chk_fail@@GLIBC_2.4
system@@GLIBC_2.2.5
get_strings
__libc_start_main@@GLIBC_2.2.5
__data_start
fprintf@@GLIBC_2.2.5
__gmon_start__
__dso_handle
_IO_stdin_used
__libc_csu_init
__bss_start
asprintf@@GLIBC_2.2.5
main
_Jv_RegisterClasses
__TMC_END__
_ITM_registerTMCloneTable
stderr@@GLIBC_2.2.5
.symtab
.strtab
.shstrtab
.interp
.note.ABI-tag
.note.gnu.build-id
.gnu.hash
.dynsym
.dynstr
.gnu.version
.gnu.version_r
.rela.dyn
.rela.plt
.init
.plt.got
.text
.fini
.rodata
.eh_frame_hdr
.eh_frame
.init_array
.fini_array
.jcr
.dynamic
.got.plt
.data
.bss
.comment


------------------
(program exited with code: 0)
Press return to continue

**I running it on Linux Mint 18. GCC version -5.3.1
Build setting - gcc -Wall   -c "%f"
Compile -       gcc -Wall   -o "%e"   "%f"**

2 ответа

Решение

Цель программы - не получать информацию от пользователя: она использует system() функция для запуска strings программа с собственным именем в качестве единственного аргумента.

Если вы работаете в среде Unix, strings Программа сканирует файлы на наличие печатаемых строк. Результат, который вы наблюдаете, более менее ожидаем: ваша программа исполняется так, как gcc содержит много строк для печати:

  • Вы можете обнаружить строковый литерал, присутствующий в исходном коде: что-то пошло не так% s.
  • многочисленные имена символов для динамического разрешения во время загрузки
  • Отладочная информация, такая как имя исходного файла: crtstuff.c
  • названия разделов, начинающиеся с .
  • Есть также несколько случайных предметов ([]A\A]A^A_, ;*3$"...) которые представляют собой просто последовательности печатных символов, присутствующих в коде исполняемого файла или двоичных данных, ошибочно интерпретируемых string как строки C, потому что они сопровождаются нулевым байтом.

В вашей программе нет места для чтения со стандартного ввода / клавиатуры. И system("strings ...") передает имя файла strings команда, так strings читает из этого файла, а не с клавиатуры.

Если вы намереваетесь читать файлы с именами файлов, переданными в вашу программу, вы должны помнить, что argv[0]это имя программы. Вам нужно посмотреть на argv[1], argv[2] и так далее.

for(int i = 1; i < argc; ++i)
    get_strings(argv[i]);
Другие вопросы по тегам