1、HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 1 -哈 尔 滨 理 工 大 学UNIX 程序设计课程设计报告题 目:单消息队列完成客户/服务器进程之间的双向通信院 系:计算机科学与技术学院 网络工程系班级学号: 姓 名: 指导教师: 郭 锦 兰 成 绩: 2013 年 12 月 23 日HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 0 - 0 -第 1 章 绪论 .11.1 选题内容 11.2 相关技术 1第 2 章 系统设计 .22.1 系统介绍 22.2 实现原理 .22.3 主要问题的解决方法
2、及关键技术 .52.4 流程图 6第 3 章 系统实现 .73.1 客户端截图 .73.2 服务器端截图 .8结 论 .9附 录 .10附录 A 核心程序代码 .10HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 0 - 0 -绪论1.1 选题内容单消息队列完成客户/服务器进程之间的双向通信1.2 相关技术随着对通信双方安全性的要求越来越高,对能使通信安全的机制需求更加迫切,UNIX 下在消息队列中嵌入加密解密算法无疑适应了潮流。UNIX为进程提供了丰富的交流方式,其中公开的交流方式有信号量、消息队列、共享内存,有名管道和文件等。采用消息队列作为通
3、信渠道可以弥补这些缺陷:1. 消息队列是一种 FIFO 的队列性结构,可以保证先送的数据先到达,后送的数据后到达,避免的插队现象。2. 消息队列将输出的信息进行了打包处理,这样就可以保证以每个消息为单位进行接收了。3. 消息队列还可以对数据分类,更可以对不同分类的数据采取不同的方式操作,这更加灵活,安全性更高。所以,利用消息队列实现客户服务器之间的加密通信是一种很安全的方式,研究单消息队列完成客户服务器之间的加密通信业显得很迫切和必要。HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 1 - 1 -系统设计2.1 系统介绍本单消息队列完成客户/服务器
4、之间的双向通信是基于消息队列的客户/服务器的通信,对通信数据信息进行有效的加密解密,实现客户 /服务器之间的双向安全通信。根据需求分析,将整个系统模块化,分为客户端和服务器端两大部分:客户端将数据加密并传输到服务端;服务器端接受数据并解密输出。2.2 流程图如下:C l i e n t发送请求消息队列请求 ?发送消息消息队列S e r v e r转存消息发送消息消息队列C h e c k ?R E JA C CF a i lS u c c消息加密消息解密HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 2 - 2 -2.3 实现原理1. 建立或返回消
5、息队列描述符进程可用系统调用 msgget 来建立或返回消息队列的描述符。该系统调用的语法格式为:int msgget(key,msgflg)key_t key;int msgflg;其中,key 是消息队列的名字;msgflg 是用户设置的标志。如果IPC_CREAT 表示系统无以 key 命名的消息队列,则建立消息队列标识符;若已存在,则返回消息队列描述符 msgid。对于系统调用,核心将搜索消息队列头标数组,确定是否有指定关键字的消息队列。若无,核心将分配一新的队列结构,并返回给用户一个消息队列描述符;否则,它只是检查消息队列的许可权之后便返回。2.消息的发送进程可用 megsnd( )
6、系统调用来发送一个消息,并将它链入消息队列的尾部。该系统调用的语法格式如下:int msgsnd(msgid,msgp,msgsz,msgflg)int msgid;struct msgbuf * msgp;int msgsz,msgflg;其中,msgid 是由 msgget 返回的消息队列描述符;msgp 指向包含这条消息的结构,该结构由如下两个成员组成:struct msgbuflong mtype; /* 消息类型 */char mtext ; /* 消息的文本 */msgsz 是 mtext 的字节长度;msgflg 规定了当无内存空间来存储消息时,进程等待还是立即返回。对于 msg
7、snd( )系统调用,核心检查消息队列描述符和许可权是否合法;HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 3 - 3 -消息长度是否超过系统规定的长度,若过长,进程睡眠等待出现足够大的空间,通过检查后,核心为消息分配消息数据区,并将消息从用户空间拷贝到消息数据区,分配消息首部,将它链入该消息队列的尾部,在消息首部填写消息类型,大小以及指向消息数据区的指针,还有修改消息队列的头标中的数据。然后唤醒在等待消息到来的队列中睡眠的进程。3. 消息的接收进程可用 msgrcv( )系统调用,从消息队列中读一条消息,语法格式为:int msgrcv(msg
8、id,msgp,msgsz,msgtyp,msgflg)int msgid,msgsz,msgflg;struct msgbuf * msgp;long msgtyp;其中,msgid,msgp,msgsz,msgflg 与 msgsnd 相似,msgtype 是规定用户想读的消息类型。对于 msgrcv( )系统调用是先由核心检查消息队列标识符和许可权,接着根据 msgtyp 分三种情况处理。(1) msgtyp=0,核心寻找消息队列中的第一个消息,并将它返回给调用进程;(2)msgtyp 为正整数,核心返回给类型的第一个消息;(3)msgtyp 为负整数,核心应在其类型值小于或等于 msg
9、typ 绝对值的所有消息中,选择类型最低的第一消息返回。如果所返回的消息的大小等于或小于用户请求,核心便将消息正文拷贝到用户区,再从队列中删除该消息,并唤醒睡眠的发送进程;如果消息比用户要求的大,则系统返回错误信息。4. 客户端发送数据的加密考虑到数据由 ASCII 码经过编程组成,通过对字符串的每个字符的 ASCII码进行加密便显得基础且高效。因此,对于客户端进程进行加密,服务器端进行解密并回传。加密通过对单个字符的 ASCII 码操作实现。5. 客户端发送数据的加密数据加密函数如下:void encode(char *buf)HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Lin
10、ux 程序设计课程设计报告- 4 - 4 -long i ;char buf11024;sscanf(buf,“%s“,buf1);memset(buf,0,sizeof(buf);for(i = 0;i #include#include#includevoid encode(char*);struct msgbuflong type;char buf1024;int msgid;struct msgbuf _msgbuf;int main()if(msgid = msgget(0x1234,0666|IPC_CREAT) #include#include#includestruct msgb
11、uflong type;char buf1024;void decode(char*);int msgid ;struct msgbuf _msgbuf;int main()if(msgid = msgget(0x1234,0666|IPC_CREAT) 0)fprintf(stderr,“msgget() failed.n“);return ;while(1)bzero(if( msgrcv(msgid,return ;fprintf(stderr,“原码:%sn“,_msgbuf.buf);HRBUST 哈尔滨理工大学计算机科学与技术学院UNIX/Linux 程序设计课程设计报告- 13
12、- 13 -decode(_msgbuf.buf);fprintf(stderr,“解密:%sn“,_msgbuf.buf);_msgbuf.type = _msgbuf.type + 10000 ;if(msgsnd(msgid,return ; /解密void decode(char *buf)int i ,n = strlen(buf);char buf14,data1024;memset(data,0,sizeof(data);for(i = 0 ;i n;)memset(buf1,0,sizeof(buf1);sscanf(buf,“%03s%s“,buf1,buf); i = i+3 ;sprintf(data,“%s%c“,data,atoi(buf1)-588);sprintf(buf,“%s“,data);