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

С под Linux

Войти
Регистрация
Восстановить пароль
 
loorke
2 / 2 / 1
Регистрация: 03.12.2016
Сообщений: 10
Записей в блоге: 1
#1

Необычное поведение при использовании каналов из <unistd.h> - C Linux

12.12.2016, 04:18. Просмотров 339. Ответов 3

Нужно написать программу на Си использующую несколько процессов. При написании столкнулся с необычной проблемой. При вызове write(pipe[1], string, strlen(stirng) в стандартный вывод подаётся содержимое строки. Сама программа выглядит вот так:
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
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#include <unistd.h>
#include <string.h>
 
 
int main(void) 
{
    int clientPipe[2];
    int* size = malloc(sizeof(int));
    char string[] = "Hello, world!\n";
 
 
    pid_t childpid;
    
    if ((childpid = fork()) == -1) {
        perror("fork");
        exit(1);
    } else if (childpid == 0) {
        /* Client */
 
        close(clientPipe[0]);
 
/*        scanf("%d", size);*/
/*        printf("%d\n", *size);*/
        /* write(clientPipe[1], size, sizeof(int)); */
        write(clientPipe[1], string, sizeof(string));
        printf("TROLOLO\n");
 
/*        time_t rawtime;*/
/*        srand(rawtime);*/
        exit(0);
    } else {
        /* Server */
    
        wait();
        close(clientPipe[1]);
        printf("Continue\n");
        *size = read(clientPipe[0], string, sizeof(string));
        printf("%s", string);
/*        read(clientPipe[0], size, sizeof(int));*/
/*        printf("%d\n", *size);*/
    }
       
    return(0);
}
Вывод при запуске следующий:
Код
TROLOLO
Hello, world!
Continue
Hello, world!
}
Почему Hello, world! выводится дважды?


И, в дополнение, когда нужно вызывать wait()? До закрытия канала и считывания или после?


Ну и раз уж зашла речь о wait(), почему при компиляции (clang source.c) я получаю следующий варнинг и как мне его не получать (гугл не помог):
Код
hello.c:37:9: warning: implicit declaration of function 'wait' is invalid in C99
      [-Wimplicit-function-declaration]
        wait();
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
12.12.2016, 04:18     Необычное поведение при использовании каналов из <unistd.h>
Посмотрите здесь:
C Linux Различное поведение при добавлении в вывод std::endl
Использование неименованных каналов, заданных явным образом C Linux
Межпотоковое взаимодействие с использованием разделяемой памяти и неименованных каналов C Linux
Странное поведение malloc C Linux
C Linux recv() и send() странное поведение
Неадекватное поведение указателя на тип char C Linux
C++ Builder Необычное поведение цикла for
Необычное поведение у цикла PHP
Необычное поведение порта
Необычное поведение &nbsp; HTML, CSS
C (СИ) Необычное поведение оператора инкремента
Android Распознавание речи и клавиатура - время ожидания - необычное поведение

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

Или воспользуйтесь поиском по форуму:
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
shvyrevvg
413 / 413 / 156
Регистрация: 12.05.2016
Сообщений: 1,137
12.12.2016, 08:02     Необычное поведение при использовании каналов из <unistd.h> #2
Не вижу
C
1
pipe(clientPipe);
Что это за wait(), где он объявлен?
Цитата Сообщение от loorke Посмотреть сообщение
гугл не помог
man pipe
qwarta
70 / 70 / 7
Регистрация: 20.11.2009
Сообщений: 234
12.12.2016, 08:06     Необычное поведение при использовании каналов из <unistd.h> #3
Почему Hello, world! выводится дважды?
Нет открытия канала: pipe()
когда нужно вызывать wait()?
Кликните здесь для просмотра всего текста
Мы хотим, чтобы родительский процесс дождался завершения дочернего процесса и вывел сообщение о том, как был завершен дочерний процесс. Первую часть этой задачи мы могли бы решить с помощью уже известного нам механизма сигналов. Каждый раз, когда дочерний процесс приостанавливается или завершается, родительский процесс получает сигнал SIGCHLD. Мы могли бы приостановить работу программы до получения этого сигнала, но таким образом мы бы знали только то, что один из дочерних процессов программы завершился, но не знали бы, ни как он завершился, ни какой именно это был процесс. Причина этого конечно, в том, что сигналы сами по себе не несут никакой дополнительной информации. Однако, в нашем распоряжении есть функция wait(2), которая приостанавливает процесс до тех пор, пока один из его дочерних процессов не будет остановлен или не завершится, после чего возвращает информацию о том, какой процесс завершился и что стало причиной его завершения. Значение, возвращаемое функцией wait() – это PID завершившегося процесса, а аргументом функции должен быть указатель на переменную status типа int. В этой переменной функция вернет дополнительные сведения о том, как завершился процесс. Вы могли подумать, что после того как мы создали два процесса с помощью fork(), не так уж важно, запускаем ли мы новую программу в дочернем или в родительском процессе, ведь разница между ними невелика. Теперь вы знаете как минимум одну причину придерживаться строгих правил, ведь родительский процесс может следить за дочерним, в то время как обратное невозможно.

Значение переменной status, в которой функция wait() передает дополнительные данные о завершившемся процессе, представляет собой маску из нескольких разных параметров. В файле <sys/wait.h> определены макросы, упрощающие «расшифровку» этой маски. Макрос WIFEXITED возвращает значение 1, если процесс завершился «добровольно», то есть в результате вызова exit() или _exit(). В этом случае с помощью макроса WEXITSTATUS можно узнать код завершения, возвращенный процессом. Макрос WIFSIGNALED возвращает 1, если выполнение процесса было завершено сигналом. Номер этого сигнала можно узнать с помощью макроса WTERMSIG. Макрос WIFSTOPPED возвращает значение 1, если выполнение процесса было приостановлено сигналом, номер которого возвращает макрос WSTOPSIG.

почему при компиляции
C
1
#include <sys/wait.h>
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
49
50
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
 
 
int main(void) 
{
    int clientPipe[2],status;
    int* size = malloc(sizeof(int));
    char string[] = "Hello, world!\n";
    pid_t childpid;
    
    pipe(clientPipe);
        if ((childpid = fork()) == -1) {
        perror("fork");
        exit(1);
    } else if (childpid == 0)                    /* Client */
            {
                close(clientPipe[0]);
                write(clientPipe[1], string, sizeof(string));
                close(clientPipe[1]);
                exit(0);
    } 
    else 
    {
        /* Server */
        
        close(clientPipe[1]);                        
        *size = read(clientPipe[0], string, sizeof(string));
        printf("%s", string);
        close(clientPipe[0]);
        if (wait(&status) == -1) 
            {
                perror("wait");
                return EXIT_FAILURE;
            }
        if (WIFEXITED(status))
            printf("Child terminated normally with exit code %i\n",WEXITSTATUS(status));
        if (WIFSIGNALED(status))
            printf("Child was terminated by a signal #%i\n", WTERMSIG(status));
        if (WCOREDUMP(status))
            printf("Child dumped core\n");
        if (WIFSTOPPED(status))
            printf("Child was stopped by a signal #%i\n", WSTOPSIG(status));
    }
    
       
    return(0);
}
drfaust
256 / 146 / 37
Регистрация: 02.10.2008
Сообщений: 530
Записей в блоге: 1
Завершенные тесты: 1
12.12.2016, 08:22     Необычное поведение при использовании каналов из <unistd.h> #4
Чёт не вижу где каналы создаются...
Yandex
Объявления
12.12.2016, 08:22     Необычное поведение при использовании каналов из <unistd.h>
Ответ Создать тему
Опции темы

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