#include <stdio.h> #include <stdlib.h> #include<sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <errno.h> #include <stropts.h> #include <time.h> #include <strings.h> #include <string.h> #include <stdio.h> #include <sys/ipc.h> #include <sys/msg.h> struct msg { long msg_types; char msg_buf[511]; }; int main(void) { int qid; int pid; int len; struct msg pmsg; pmsg.msg_types = getpid(); sprintf(pmsg.msg_buf, "hello!this is :%d\n\0", getpid()); len = strlen(pmsg.msg_buf); if ((qid = msgget(IPC_PRIVATE, IPC_CREAT | 0666)) < 0) { perror("msgget"); exit(1); } if ((msgsnd(qid, &pmsg, len, 0)) < 0) { perror("magsn"); exit(1); } printf("successfully send a message to the queue: %d \n", qid); exit(0); }的运行结果
时间: 2023-09-11 13:09:26 浏览: 118
该程序是一个使用消息队列进行进程间通信的示例程序,通过msgget()函数创建一个消息队列,通过msgsnd()函数向消息队列中发送一条消息。
运行结果可能如下:
successfully send a message to the queue: 123456
解释如下:
1.程序首先定义了一个消息结构体msg,并初始化了其中的消息类型msg_types和消息内容msg_buf。
2.程序调用msgget()函数创建一个新的消息队列,将返回的队列ID存储在变量qid中。
3.程序调用msgsnd()函数向消息队列中发送一条消息,消息内容为msg_buf中的字符串。
4.程序输出"successfully send a message to the queue: qid",表示消息发送成功。
需要注意的是,程序中将消息类型msg_types设置为了当前进程的PID,这样在接收消息时可以根据PID来获取对应的消息。同时,程序中使用了strlen()函数来获取消息内容的长度,这是为了确保发送的消息长度不超过队列的最大长度。
相关问题
例 2:命名管道通信实例 分别编写读写进程的程序 write.c 和 read.c,两个程序之一在当前目录下创建一个 命名管道“mypipe”,然后 write 向管道写数据,read 从管道读数据,两个进程可 任意顺序运行。 编写 write.c: //write.c #include<stdio.h> #include <stdlib.h> #include <sys/types.h> #include <sys/stat.h> #include <error.h> #include <fcntl.h> #include <unistd.h> #define N 256 int main(){ char buf[N]; int fd= open("./mypipe",O_WRONLY|O_CREAT,0666); if(fd!=-1) { printf("FIFO file is opened\n"); } else { perror("open failed"); exit(0); } printf("please input string\n"); scanf("%s",buf); getchar(); if ( write(fd,buf,sizeof(buf))!=-1 ) printf("write successful\n"); else perror("write failed:"); exit(EXIT_SUCCESS); } 编写 read.c: //read.c #include<stdio.h> #include <stdlib.h> #include <sys/types.h> #include <sys/stat.h> #include <error.h> #include <fcntl.h> #include <unistd.h> #define N 256 int main(){ int fd= open("./mypipe",O_RDONLY|O_CREAT,0666); char buf[N]; if(fd!=-1) { printf("FIFO file is opened\n"); } else { perror("open failed"); exit(0); } if ( read(fd,buf,N )!=-1 ) printf("I received data %s\n",buf); else perror("read error:"); exit(EXIT_SUCCESS); } 运行方式:打开 2 个终端,分别运行读写进程。 请完成以下练习与回答问题: 练习 1:改写本例,使得写进程可以不断的向管道文件写,读进程可以不断的读, 思考如何控制读写顺序。 练习 2:本例中用于管道通信的是一个普通文件,请用 mkfifo 命令或 mkfifo( )函 数创建一个标准管道文件改写本例,查看一下通过管道文件不断读写有什么不同? 问题 1:请说明匿名管道与命名管道在创建方式上有何不同?为什么说匿名管道 只能用于有亲缘关系的进程间进行通信?
练习 1:
可以在写进程和读进程中都用 while 循环,不断地读写数据。为了控制读写顺序,可以在读进程中使用 sleep() 函数来暂停一段时间,等待写进程写入数据。
练习 2:
可以使用 mkfifo() 函数创建一个标准管道文件,具体操作可以参考以下代码:
```c
mkfifo("mypipe", 0666);
```
使用标准管道文件和命名管道文件进行通信的方式是一样的,不同的只是创建方式。通过标准管道文件进行通信时,需要先手动创建管道文件,而命名管道则可以在程序中动态创建。
问题 1:
匿名管道和命名管道在创建方式上的不同在于,匿名管道需要通过 pipe() 函数创建,而命名管道需要通过 mkfifo() 函数创建。
匿名管道只能用于有亲缘关系的进程间进行通信,因为它只能在创建它的进程及其子进程之间使用。而命名管道则可以被多个进程共享使用,因为它是通过文件系统中的文件来实现的。
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <dirent.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <pwd.h> #include <grp.h> #include <time.h> void shell_ls_l(char *file,struct stat st) { char limi
ted[11]; // 文件权限 mode_t mode = st.st_mode; limitted[0] = (S_ISDIR(mode)) ? "d" : "-"; limitted[1] = (mode & S_IRUSR) ? "r" : "-"; limitted[2] = (mode & S_IWUSR) ? "w" : "-"; limitted[3] = (mode & S_IXUSR) ? "x" : "-"; limitted[4] = (mode & S_IRGRP) ? "r" : "-"; limitted[5] = (mode & S_IWGRP) ? "w" : "-"; limitted[6] = (mode & S_IXGRP) ? "x" : "-"; limitted[7] = (mode & S_IROTH) ? "r" : "-"; limitted[8] = (mode & S_IWOTH) ? "w" : "-"; limitted[9] = (mode & S_IXOTH) ? "x" : "-"; limitted[10] = '\0'; printf("%s ", limitted); // 链接数、所有者、所在组、文件大小、创建时间、文件名 printf("%ld ", (long)st.st_nlink); printf("%s ", getpwuid(st.st_uid)->pw_name); printf("%s ", getgrgid(st.st_gid)->gr_name); printf("%lld ", (long long)st.st_size); char time_buf[20]; strftime(time_buf, 20, "%b %d %H:%M", localtime(&st.st_ctime)); printf("%s ", time_buf); printf("%s\n", file); } int main(int argc, char *argv[]) { DIR *dir; struct dirent *ptr; struct stat st; char path[256]; if (argc == 1) { getcwd(path, 256); dir = opendir(path); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; stat(ptr->d_name, &st); shell_ls_l(ptr->d_name, st); } } else if (argc == 2) { if (strcmp(argv[1], "-l") == 0) { getcwd(path, 256); dir = opendir(path); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; stat(ptr->d_name, &st); shell_ls_l(ptr->d_name, st); } } else { dir = opendir(argv[1]); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; sprintf(path, "%s/%s", argv[1], ptr->d_name); stat(path, &st); shell_ls_l(ptr->d_name, st); } } } else if (argc == 3) { if (strcmp(argv[1], "-l") == 0) { dir = opendir(argv[2]); while ((ptr = readdir(dir)) != NULL) { if (ptr->d_name[0] == '.') continue; sprintf(path, "%s/%s", argv[2], ptr->d_name); stat(path, &st); shell_ls_l(ptr->d_name, st); } } } return 0; }
这是一个简单的实现Linux命令ls -l的程序,可以列出文件的权限、链接数、所有者、所在组、文件大小、创建时间、文件名等信息。
阅读全文