Задачка про Юникс и Си
Mar. 8th, 2008 11:47 pmВот задача про Юникс и Си, её придумал пользователь
avysk, а я слегка модифицировал.
Написать на Си коротенькую программу, которая при вызове через конвейер выдаёт фразу "Hello, world!\n", а при вызове с командной строки выдаёт ту же строку, но не один, а два раза.
Короче, вот так:
$ my_prog
Hello, world!
Hello, world!
$ my_prog | cat -
Hello, world!
$
Конечно есть функция isatty, с помощью которой можно просто проверить, запущена ли программа с терминала, вот так:
if(isatty(1))...else...;
Поэтому дополнительное условие: этой функцией (или подобными ей) пользоваться нельзя.
И вот что интересно - пользователь
ilya_dogolazky запостил очень похожую задачу.
Похожую, да решающуюся несколько иначе, хоть и непросто найти отличия в условии.
Написать на Си коротенькую программу, которая при вызове через конвейер выдаёт фразу "Hello, world!\n", а при вызове с командной строки выдаёт ту же строку, но не один, а два раза.
Короче, вот так:
$ my_prog
Hello, world!
Hello, world!
$ my_prog | cat -
Hello, world!
$
Конечно есть функция isatty, с помощью которой можно просто проверить, запущена ли программа с терминала, вот так:
if(isatty(1))...else...;
Поэтому дополнительное условие: этой функцией (или подобными ей) пользоваться нельзя.
И вот что интересно - пользователь
Похожую, да решающуюся несколько иначе, хоть и непросто найти отличия в условии.
no subject
on 2008-03-08 10:08 pm (UTC)no subject
on 2008-03-08 10:49 pm (UTC)Ответная задачка:
Bookintosh:user$ ./my_prog
Hello world 1!
Hello world 2!
Bookintosh:user$ ./my_prog | cat -
Hello world 2!
Hello world 1!
no subject
on 2008-03-09 12:00 am (UTC)printf("Hello, world 1\n") ;
fprintf(stderr, "Hello, world 2\n") ;
то тут вот какое дело: в обеих исходных задачах вывод происходит в stdout, а stderr вообще не задействован: можно смело ко всем вызовам приписать 2>/dev/null и ничего не изменится.
А в каком смысле "bad file descriptor"?
no subject
on 2008-03-09 12:15 am (UTC)В смысле что
fd = fcntl(STDOUT_FILENO, F_DUPFD, 0);
write( fd, hello, strlen( hello ) );
in = read( STDIN_FILENO, rx, 0 );
if( in != -1 )
write( fd, hello, strlen( hello ) );
ну и в обратную сторону раскручивается с тем же успехом. Я не помню как это фифект оффициально называется (с чтением STDOUT), но вот такая вот байда.
Если вы имеете в виду
printf("Hello, world 1\n") ;
fprintf(stderr, "Hello, world 2\n") ;
Совершенно верно. Но это была ответная задачка.
no subject
on 2008-03-09 01:10 am (UTC)Но это не то :-)
no subject
on 2008-03-08 10:49 pm (UTC)no subject
on 2008-03-08 10:58 pm (UTC)no subject
on 2008-03-08 11:08 pm (UTC)а какую — не скажу из соображений симметрии ;)
no subject
on 2008-03-08 11:11 pm (UTC)no subject
on 2008-03-08 11:25 pm (UTC)no subject
on 2008-03-08 11:38 pm (UTC)no subject
on 2008-03-08 11:52 pm (UTC)no subject
on 2008-03-09 01:07 am (UTC)no subject
on 2008-03-09 12:12 am (UTC)no subject
on 2008-03-09 01:02 am (UTC)no subject
on 2008-03-09 01:19 am (UTC)no subject
on 2008-03-09 01:26 am (UTC)no subject
on 2008-03-09 01:32 am (UTC)no subject
on 2008-03-09 01:38 am (UTC)no subject
on 2008-03-09 02:00 am (UTC)А, здорово! Я вот немножко подсократил, так что получилось 4 «существенных» строчки - http://codepad.org/hh15AzmC
Если хотите можете подумать, как обойтись лишь двумя принтф"ами и ещё одним дополнительным вызовом некой функции (сразу скажу, что получится не слишком переносимо, но просто считайте, что запускаем под линуксом и всё тогда будет окей).
[если Вы ещё не спите и если не сложно, потрите пожалуйста коммент про волшебное слово на букву «с», а то с утра народ подтянется, его увидит и расстроится. А уж в ссылки с готовыми решениями никто случайно не ткнёт :-) ]
no subject
on 2008-03-09 02:02 am (UTC)no subject
on 2008-03-09 02:04 am (UTC)no subject
on 2008-03-09 09:56 am (UTC)no subject
on 2008-03-09 09:20 am (UTC)no subject
on 2008-03-09 10:48 am (UTC)no subject
on 2008-03-09 10:50 am (UTC)Почитать текст со всеми комментариями можно тут (http://deep-water.ru/?http://dimrub.livejournal.com/646117.html)
Это Ваш 5-й ТОПовый пост за последний год (http://deep-water.ru/top/). Посмотреть статистику автора можно тут (http://deep-water.ru/top/info.php?id=3361).
Этот "бот не имеет отношения к Яндексу" © НадежныйИсточник
no subject
on 2008-03-09 11:00 am (UTC)no subject
on 2008-03-10 05:57 am (UTC)#include <stdlib.h>
#include <unistd.h>
int main(void) {
printf("Hello world!\n");
fork();
return EXIT_SUCCESS;
}
no subject
on 2008-03-10 11:19 am (UTC)no subject
on 2008-03-10 12:21 pm (UTC)Ну, можно смухлевать и воспользоваться ttyname, если ограничение касается только isatty. Можно позвать ioctl(STDOUT_FILENO, TCGET ...), но это тоже жульничество, ибо isatty - это просто обертка для этого ioctl.
Можно попробовать почитать из STDOUT_FILENO в неблокируемом режиме: если это перенаправление, то будет EBADF, а если будет EAGAIN, то перенаправления не было. Но, боюсь, это не переносимо. На линуксе это так, а на других системах сейчас проверить нет возможности.
Можно позвать fstat, который все расскажет, но это не интересно.
В общем, решений масса, но "прикольное" что-то не придумывается.
no subject
on 2008-03-10 12:30 pm (UTC)