- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有 2 个进程,父进程和子进程。父进程应该等待用户插入一些字符串,然后它应该通过管道将这个字符串发送给子进程。我已经成功地做到了这一点,但我想要实现的是检查管道是否为空,如果它为空 60 秒,那么子进程应该向其父进程发送一个信号。
到目前为止我的代码:
int main(void)
{
int fd[2], nbytes;
pid_t childpid;
char string[100];
char readbuffer[80];
pipe(fd);
if((childpid = fork()) == -1)
{
perror("fork");
exit(1);
}
else if(childpid > 0) //parent process
{
close(fd[0]);
printf ("Insert a string: ");
scanf("%s",string);
/* Send "string" through the output side of pipe */
write(fd[1], string, (strlen(string)+1));
exit(0);
}
else // child process
{
close(fd[1]);
// check if pipe is empty for 60 seconds
nbytes = read(fd[0], readbuffer, sizeof(readbuffer));
printf("Received string: %s", readbuffer);
}
return(0);
}
最佳答案
这是一个示例,其中父级 sleep()
随机数秒,然后通过管道 write()
s 给子级。如果子进程在5.5秒内没有收到消息,则发送SIGUSR1
给父进程,父进程不写消息。这将重复指定的次数。
请注意, child 等待 5.5 秒,而不是 5 秒,因为我们在父级中使用 sleep()
,这需要整整几秒。这(通常)会留下足够的时间间隔,以避免父 sleep()
和子等待 5 秒,并且它们的电线在中间交叉的情况。我们可以在父级中使用 nanosleep()
,让子级等待整数秒。
注意事项:
我们可以在子进程中使用带有超时的select()
来等待指定的时间量。 select()
将返回就绪文件描述符的数量,因此如果我们只检查管道并返回 0
,我们就知道管道中没有任何内容可供读取。
我们可以在父级中使用sleep()
来测试它。 sleep()
如果它在请求的时间内休眠则返回 0,如果它被信号中断则返回非零,因此我们可以检查返回值以确定是哪种情况。对于您的特定情况,如果您希望父级从用户那里获取输入,read()
将返回 -1
并将 errno
设置为 EINTR
如果它正在等待用户输入并接收到信号,那么您可以通过这种方式检测超时。当然,在这个特定的用例中, parent 更容易在 60 秒超时时在 STDIN_FILENO
上调用 select()
本身,而不是让 child 等待管道并发送信号。
在 child 中,我们必须在每次循环中FD_ZERO
fd_set
并重新填充struct timeval
,因为select()
可能会修改它们。
我们需要为 SIGUSR1
注册一个信号处理程序(它什么都不做)以避免进程在收到信号时被终止。我们不需要信号处理程序做任何事情,因为 sleep()
会在它被信号中断时通过返回值告诉我们,这就是我们需要的所有信息,就在这里。
代码:
#define _XOPEN_SOURCE 500
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <signal.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/select.h>
#define READ_END 0
#define WRITE_END 1
#define SEC_THRESHOLD 5
#define LOOP_TIMES 5
#define BUFFER_SIZE 512
void handler(int signum)
{
(void) signum; // Ignore unused argument
}
int main(void)
{
int p[2];
if ( pipe(p) == -1 ) {
perror("pipe() failed");
exit(EXIT_FAILURE);
}
pid_t pid = fork();
if ( pid == -1 ) {
perror("fork() failed");
exit(EXIT_FAILURE);
}
else if ( pid == 0 ) {
if ( close(p[WRITE_END]) == -1 ) {
perror("failed to close pipe end in child");
exit(EXIT_FAILURE);
}
for ( int i = 0; i < LOOP_TIMES; ++i ) {
fd_set fds;
FD_ZERO(&fds);
FD_SET(p[READ_END], &fds);
struct timeval timeout;
timeout.tv_sec = SEC_THRESHOLD;
timeout.tv_usec = 500000;
printf("Loop %d: child waiting for %d.5 seconds.\n",
i, SEC_THRESHOLD);
int status = select(p[READ_END] + 1, &fds, NULL, NULL, &timeout);
if ( status == -1 ) {
perror("select() failed in child");
exit(EXIT_FAILURE);
}
else if ( status == 0 ) {
printf("Loop %d: timed out in child, sending signal.\n", i);
kill(getppid(), SIGUSR1);
}
else {
char buffer[BUFFER_SIZE] = {0};
if ( read(p[READ_END], buffer, BUFFER_SIZE - 1) == -1 ) {
perror("read() failed in child");
exit(EXIT_FAILURE);
}
printf("Loop %d: child read: [%s]\n", i, buffer);
}
}
if ( close(p[READ_END]) == -1 ) {
perror("failed to close read end of pipe in child");
exit(EXIT_FAILURE);
}
exit(EXIT_SUCCESS);
}
else {
if ( close(p[READ_END]) == -1 ) {
perror("failed to close pipe end in parent");
exit(EXIT_FAILURE);
}
struct sigaction sa;
sa.sa_handler = handler;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
sigaction(SIGUSR1, &sa, NULL);
srand((unsigned) time(NULL));
for ( int i = 0; i < LOOP_TIMES; ++i ) {
const char * msg = "Message to child";
const int wait_time = rand() % 6 + 3;
printf("Loop %d: parent waiting for %d seconds...\n", i, wait_time);
int status = sleep(wait_time);
if ( status == 0 ) {
printf("Loop %d: parent sending message to child.\n", i);
if ( write(p[WRITE_END], msg, strlen(msg)) == -1 ) {
perror("write() error in parent");
exit(EXIT_FAILURE);
}
}
else {
printf("Loop %d: parent interrupted by signal.\n", i);
}
}
if ( close(p[WRITE_END]) == -1 ) {
perror("failed to close write end of pipe in parent");
exit(EXIT_FAILURE);
}
}
if ( waitpid(pid, NULL, 0) == -1 ) {
perror("waitpid() failed");
exit(EXIT_FAILURE);
}
return EXIT_SUCCESS;
}
和示例输出 session :
paul@horus:~/src/sandbox$ ./sigpipe
Loop 0: parent waiting for 7 seconds...
Loop 0: child waiting for 5.5 seconds.
Loop 0: timed out in child, sending signal.
Loop 1: child waiting for 5.5 seconds.
Loop 0: parent interrupted by signal.
Loop 1: parent waiting for 7 seconds...
Loop 1: timed out in child, sending signal.
Loop 2: child waiting for 5.5 seconds.
Loop 1: parent interrupted by signal.
Loop 2: parent waiting for 6 seconds...
Loop 2: timed out in child, sending signal.
Loop 3: child waiting for 5.5 seconds.
Loop 2: parent interrupted by signal.
Loop 3: parent waiting for 5 seconds...
Loop 3: parent sending message to child.
Loop 4: parent waiting for 3 seconds...
Loop 3: child read: [Message to child]
Loop 4: child waiting for 5.5 seconds.
Loop 4: parent sending message to child.
Loop 4: child read: [Message to child]
paul@horus:~/src/sandbox$
因此我们可以看到,如果子级达到 5.5 秒的超时时间,父级的每次迭代都会被中断,而在所有其他情况下成功发送消息。
关于检查管道是否为空,UNIX,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34374000/
正如标题所暗示的那样,我无法弄清楚 Unix 内核如何将逻辑文件偏移量转换为逻辑块号,然后从 i-node 中检索它。 作为引用,我要求对 Maurice J. Bach 在“UNIX 操作系统的设计
关闭。这个问题是off-topic .它目前不接受答案。 想改善这个问题吗? Update the question所以它是 on-topic对于堆栈溢出。 8年前关闭。 Improve this q
我在互联网上做了一些研究,但仍然很困惑。 UNIX 时间是像 GMT/UTC 那样的通用时间还是像本地时间一样因地而异? 我知道 UNIX 时间是从 1970 年 1 月 1 日格林威治标准时间 00
您如何评估 Unix 系统管理员。 Unix 管理员应该具备哪些编程能力? 最佳答案 我用于快速过滤器的一些: 什么是 fork 炸弹,它是好是坏? 给我一个单行命令,计算日志文件中有多少行从昨天的日
谁能告诉我字典文本文件在 UNIX 系统上的位置?或者我在哪里可以获得一个好的字典文本文件?我目前一直在使用来自 SUN 的文本文件,但它包含不带句点的缩写(否则我可以删除它们)。有人能指出我正确的方
在我的 Ubuntu 12 vps 上,我正在运行一个完整的比特币节点。当我第一次启动它时,它使用了大约 700mb 的内存。如果我 24 小时后回来 (free -m) 将如下所示: total
我想编写一个 unix/linux 程序,它将使用一个配置文件。 我的问题是,我应该把文件的位置放在哪里? 我可以将位置(如 /etc )“硬编码”到程序本身中。 但是,我希望它,如果没有权限的用户可
在UNIX脚本编程中,cat是可以将2个文件组合在一起的命令: cat file1 file2 > file3 通过合并前两个生成第三个。 另外,cat可以与管道一起使用: cat file1 | t
我有一个通过 ssh 连接到外部机器的终端,并且有一个进程在其中运行。 是否可以将执行移到后台,以便我可以关闭 ssh 连接而无需终止它?如果是这样怎么办? 最佳答案 按 control + Z,这将
我正在试验我自己的 BSD 或 Linux 发行版。我想以对最终用户有意义的方式组织系统文件。我希望他们能够访问系统,而不会出现 *nixes 留下的所有文件困惑。 有没有办法在不丢失动态链接的情况下
这条评论让我感到困惑:“kill -l 通常会列出所有信号”。我认为信号意味着量化的能量。 [已添加] 请澄清 Unix 中的(计算)信号和物理信号。它们是完全不同的概念吗? [已添加] 范式之间是否
fuser 命令让我知道哪些进程正在使用文件或目录。 我正在寻找相反的命令:让我知道进程正在使用哪些文件。 更新 忘了说它是针对 Solaris 系统的。 最佳答案 lsof -p 来自 here
如果我有一个叫做“orange”的词,我如何将它拆分成单独的字符。 我的输出应该是: o r a n g e 最佳答案 echo orange | fold -w 1 输出 o r a n g e 关
和有什么区别工作和一个 流程在 Unix 中?你能举个例子吗? 最佳答案 作业是由 shell 启动的进程。 shell 在作业表中跟踪这些。作业命令显示事件后台进程的列表。他们得到一个 jobspe
unix 如何处理带空格和参数的完整路径名? 在 Windows 中,我们引用路径并在其后添加命令行参数,在 unix 中如何? "c:\foo folder with space\foo.exe"
我必须合并具有相同标题的多个CSV文件。 我必须保留第一个文件的 header ,并删除所有其他文件的 header ,然后合并它们并创建一个主文件。 文件1: Id,city,name ,locat
我需要在两个字段上加入两个文件。但是,即使连接失败,我也应该检索文件 1 中的所有值,就像左外连接一样。 文件 1: 01|a|jack|d 02|b|ron|c 03|d|tom|e 文件2: 01
在 Solaris, HP-UX 上获取进程大小的正确方法是什么?和 AIX ?我们应该使用 top或 ps -o vsz或者是其他东西? 最佳答案 vsize的确切定义, rss , rprvt ,
我在文件上使用了“touch”,更新了文件的时间戳,但父目录的时间戳没有改变。但是,(如预期)当我在父目录中创建新文件时,该目录的时间戳确实发生了变化。 类 UNIX 操作系统(特别是 AIX)使用什
一般来说,当我们从多个进程向 UNIX 中的文件追加内容时,我们可以认为什么是理所当然的?是否有可能丢失数据(一个进程覆盖另一个进程的更改)?数据有可能被破坏吗? (例如,每个进程都将每个追加一行追加
我是一名优秀的程序员,十分优秀!