操作系统:进程间通信

#include<unistd.h>
#include<stdlib.h>
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/sem.h>

#ifndef _SEMUN_H          //条件编译,即若SEMUN在头文件中没有被定义,就进行下面的编译
#define _SEMUN_H
union semun{    
   int val;                          //设置semnum指定的信号量的值
   struct semid_ds *buf;   
   unsigned short int *array;
   struct seminfo *_buf;
};
#endif


static int set_semvalue(void);
static void del_semvalue(void);
static int semaphore_p(void);
static int semaphore_v(void);

static int sem_id;

int main(int argc,char *argv[]){
   int i;
   int pause_time;
   char *op_in="First in/n";
   char *op_out="First out/n";

   srand((unsigned int)getpid());                    //以getpid()为随机序列的种子
   sem_id=semget((key_t)1234,1,0666|IPC_CREAT);   /*为一个新创建的集合返回标识符,
                                             或者返回具有相同键值的已存在信号集的标识符*/

   if(argc>1){
      if(!set_semvalue()){                         //信号初始化失败
         fprintf(stderr,"Failed to initialize semaphore/n");
      }
      op_in="Second in/n";
      op_out="Second out/n";
      sleep(2);
   }
   for(i=0;i<5;i++){
      if(!semaphore_p())
         exit(EXIT_FAILURE);
      printf("%s",op_in);
      fflush(stdout);
      pause_time=rand()%3;
      sleep(pause_time);
      printf("%s",op_out);
      fflush(stdout);
  
      if(!semaphore_v())
         exit(EXIT_FAILURE);
      pause_time=rand()%2;
      sleep(pause_time);
   }
   printf("/n%d-finished/n",getpid());
  
   if(argc>1){
      sleep(10);
      del_semvalue();
   }
   exit(EXIT_SUCCESS);
}


static int set_semvalue(void){     
   union semun sem_union;

   sem_union.val=1;
   if(semctl(sem_id,0,SETVAL,sem_union)==-1)     /* 根据sem_union.val设置semmun指定的信号量的值,
                                                错误返回-1,成功返回正数 */
      return(1);                               //前面条件成立,即返回-1,则返回调用函数中的值为1
}

static void del_semvalue(void){               
   union semun sem_union;
   if(semctl(sem_id,0,IPC_RMID,sem_union)==-1)   /* 从系统中删除标识符为sem_id的信号集,
                                               错误返回-1,正确返回正数 */
      fprintf(stderr,"Failed to delete semaphore/n");
}

static int semaphore_p(void){
   struct sembuf sem_b;        //sembuf结构在Linux/sem.h中有定义
 
   sem_b.sem_num=0;          //信号在数组中的索引
   sem_b.sem_op=-1;          /* 这里为负数,相当于P操作,从信号量中减去sem_op的绝对值(1) */
                              
                              
   sem_b.sem_flg=SEM_UNDO; /*操作标志为SEM_UNDO,当一个进程改变了信号量的值进入临界区后,
                           但因为其他原因没有退出临界区的情况下,会造成信号量永远不能释放,
                         此时,系统用SEM_UNDO实施还原操作,信号量还原为进程操作前的状态 */
   if(semop(sem_id,&sem_b,1)==-1){     //P操作发生错误,返回-1,条件成立;否则条件不成立
      fprintf(stderr,"semaphore_p failed");
      return(0);               //上面的条件成立,即发生错误,返回0给调用函数
   }
}         //luojiazhen corrected it 05年12月20日


static int semaphore_v(void){     
   struct sembuf sem_b;

   sem_b.sem_num=0;
   sem_b.sem_op=1;            /* 这里为正数,相当于V操作,在信号量中增加sem_op的值(1),
                                 这意味着该进程从临界区返回,释放临界资源 */        
   sem_b.sem_flg=SEM_UNDO;
     
   if(semop(sem_id,&sem_b,1)==-1){      //V操作发生错误,返回-1,条件成立;否则条件不成立
      fprintf(stderr,"semaphore_v failed");
      return(0);
   }
   return(1);
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: Unix网络编程卷2:进程间通信PDF是一本非常有用的书籍,它涵盖了关于进程间通信的所有关键知识。 进程间通信是指进程之间交换数据或信息的过程,这对于理解操作系统以及网络编程非常重要。本书不仅讲解了进程通信的基础知识,还深入解释了信号、管道、消息队列、共享内存等高级通信方法。 在本书中,作者详细介绍了如何使用各种系统调用和库函数实现不同类型的进程间通信。读者将学习如何在不同进程之间共享文件句柄,如何创建匿名和命名管道,以及如何使用信号和信号处理程序等。 此外,本书还说明了如何以面向对象的方式编写并发程序。作者展示了C++ STL标准库和Boost库的使用方法,这些工具可以帮助程序员编写更高效的并发程序。 总之,Unix网络编程卷2:进程间通信PDF是一本非常有用的书籍,可以通过实例和详细解释帮助读者更好地理解进程通信的概念和技术。 ### 回答2: 《Unix网络编程 卷2:进程间通信》是一本经典的计算机网络编程书籍,主要讲解了在UNIX环境下进程之间如何进行通信,并介绍了常用的进程间通信机制和技术。 该书包含了进程间通信的基本概念和理论知识,从分析进程地址空间、进程控制、信号处理、进程间通信等多个方面详细阐述了进程间通信的各种实现方式,并通过实际的例子和代码提供了丰富的实践经验。同时,该书还包含了大量的参考文献和附录,方便读者深入学习和进一步研究进程间通信技术。 该书涵盖的主要内容包括UNIX进程模型、基本进程管理、进程资源和限制、信号、管道、消息队列、共享内存、信号量、套接字、RPC、XSI IPC等多个进程间通信机制和技术。其中,对于常用的进程间通信方式如管道、消息队列、共享内存、信号量等都进行了详细的介绍。同时,该书还介绍了进程间通信的高级技术,如分布式进程间通信(RPC)和XSI IPC等,帮助读者更好地实现进程间的通信。 总之,《Unix网络编程 卷2:进程间通信》是一本非常重要的计算机网络编程参考书籍,对于了解UNIX进程模型、深入理解进程间通信技术以及开发UNIX网络应用程序有很大的帮助。该书不仅适合计算机专业的学生和研究人员,也适合从事UNIX网络编程工作的程序员和工程师阅读。 ### 回答3: 《UNIX网络编程 卷2:进程间通信》是由W.Richard Stevens和Stephen A. Rago合作编写的,是一本关于UNIX如何实现进程间通信的技术指南,是一本精华之作。该书的主要内容包括:管道、FIFO、消息队列、信号量、共享内存以及套接字等多种进程间通信方式。同时,书中也介绍了如何在不同的进程间进行信息交换、如何保证进程的同步性和互斥性以及如何利用各种进程间通信工具实现并发编程。 该书在介绍管道、FIFO、消息队列等进程间通信机制时,都给出了详细的代码实现,并给出了该技术的优缺点以及适用场景。同时,书中还提供了丰富的案例分析,让读者能通过实践更好地理解和掌握这些技术。此外,书中还介绍了一些进程和线程相关的基础知识,如进程的创建、精灵进程、线程的创建、线程的同步等,这些知识为读者更好地理解进程间通信技术提供了背景和基础。 总的来说,该书是一本深入浅出、全面系统的进程间通信技术指南,它通过具体的代码实现和案例分析,使得读者能在实践中更好地理解和掌握这些技术,并能够开发出高效可靠的应用程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

路漫漫其修远.

你的鼓励是我寻找真相的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值