Linux下的简单聊天工具.doc

上传人:豆**** 文档编号:23970627 上传时间:2022-07-03 格式:DOC 页数:28 大小:625.50KB
返回 下载 相关 举报
Linux下的简单聊天工具.doc_第1页
第1页 / 共28页
Linux下的简单聊天工具.doc_第2页
第2页 / 共28页
点击查看更多>>
资源描述

《Linux下的简单聊天工具.doc》由会员分享,可在线阅读,更多相关《Linux下的简单聊天工具.doc(28页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。

1、Four short words sum up what has lifted most successful individuals above the crowd: a little bit more.-author-dateLinux下的简单聊天工具概述Abstract概述AbstractLinux作为免费开源操作系统广泛应用于企业。Linux与UNIX有着深厚的渊源,而UNIX系统可谓“坚如磐石”,其稳定性受到广泛赞誉。近年来,越来越多的就业岗位被提供给了Linux开发人员。IM即时通讯软件是近年来流行的通信方式,企业、个人等都在更多地使用IM进行沟通。通过计算机网络,信息的传递变得十

2、分方便。并非所有IM软件都要想腾讯QQ那样复杂、庞大,作为内部沟通,只需功能齐全、操作方便即可。作为课程设计作品,在Linux下开发C/S式IM软件,对于熟悉Linux、复习计算机网络、进一步学习软件开发都有十分积极的意义。As is open source software, Linux has been widely used in corporations. These years, more and more jobs are offered to Linux developers, for Linux spoken highly of for its stability as UNI

3、X. IM software has become the most widely used communication software both for business use and for personal use. Through internetworks - including the Internet, Enterprise LAN, VPNs - information can be exchanged more conveniently and immediately. However, not all IM of C/S model is that complex as

4、 Tencent QQ. IM software inside an Enterprise LAN should be lighter and easier. As a product design of Linux, developing an IM software is helpful to reviewing key points of Linux, Internet, and software developing. -目 录第1章 设计任务1.1. 设计任务概述1.1.1. 课题名称Linux下的IM网络聊天软件1.1.2. 内容摘要随着嵌入式在生活中的应用越来越广泛,嵌入式LIN

5、UX下的Socket网络编程也越来越热。为了加强对在Linux系统下进行编程的学习和实践,运用计算机网络、软件工程等知识,我们选择了Linux下网络聊天工具作为课设题目。本系统主要功能在于实现多个客户端一与服务器端之问的信息传递与文件传输功能。系统主要实现4人聊天室功能:注册与登录系统,公聊,私聊,文件传输。应用LINUX下的Socket网络编程并使用TCP实现简单聊天程序。编程过程中不断用各种网络调试工具进行调试程序,最终做成了简单的在Linux下的聊天工具。关键词:Linux Socket IM软件1.2. 开发环境1.2.1. 硬件环境Intel x86 CPU架构(主频1G Hz以上)

6、512MB DDR2内存支持800*600以上分辨率、16色以上的显示设备1.2.2. 软件环境Red Hat Enterprise Linux(或Fedora)VIM编译器1.3. 要求在Linux下实现网络聊天,包括公聊、一对多私聊等功能。实现客户端之间经网络传输文件。能保存聊天记录。第2章 设计方案2.1. 系统结构2.1.1. 结构结构组成:注册、登录、聊天、文件传输。2.1.2. 结构图聊天系统分为五大模块2.2. 流程与模块2.2.1. 执行流程2.2.2. 各功能需要调用的模块模块需求注册登录聊天文件传输用户注册用户登录系统广播上下线提示用户私聊文件传输强制下线第3章 实现原理3

7、.1. 注册、登录模块3.1.1. 服务器端服务器端建立好socket,等待连接,当有客户端连接服务器的时候,服务器接收连接,并接受客户端发送过来的消息,恨据接收到的结构体所携带的协议来做相应的功能:服务器端启动后如图 31所示。图 31l、注册:如果协议为reg,则为客户端注册,首先将发送过来的结构体,提取用户名和密码,然后需要对川户名合法性检验,验证之后如果用户名合法则将川户信已保存到文件中,合法性的规则包括川户名不能重复和不能使用all等协议作为川户名,并且川户名和密码都不能为空。如果注册成功,服务器端发送一个消息给注册的客户端,同样将消息保存在一个结构休里如果失败,也给客户端发送一个消

8、息如“您榆入的用户名小能为akk”或者“用户名XX已经存在”。注册结果如图 32所示。图 322、登录:如果协议为login,则将用户名和密码信息提取,再遍历存放用户信息文件里的用户名和密码直到验证成功为止,如果验证成功则对所有在线的川户发送一条消息:“提示XX用户登录成功”;如果失败则只给登阳失败的客户端提示登录失败,并给出原囚,如“用户名不存在”或者“用户名或者密码输入错误”,并跳转到相应的代码执行其他功能,成功则等特发送客户端消息,失败则关闭Socket并结束线程,如图 33所示:图 333、监听和踢出客户端:通过六行和修改绑定的Socket和在线用户队列实现查看和踢出在线用户,踢出用户

9、后向被踢出用户发送相关信息,如图 34、图 35所示。图 34图 353.1.2. 客户端客户端的输入和消息的显示要使用2个终端,一个是Client,一个是Display。Client终端为输入的界面。在这个界面里,新建一个线程来接受服务器端发来的消息,再添加时间信息,并将这些信息写入文件,然后给Display进程发送一个消息,Display进程接到消息,就去读取文件并将这些数拟显示在Display终端。打开客户端Display终端界而,用lseek将内部指t阵行向文件末尾,等待Client终端里的线秤将消息写入文件。一旦有消息过来就去文件里读取数据井打印在Display终端。打开客户端Cli

10、ent终端界而,有3个菜单,包括注册、登录、退出。选择相应项即可进幻相关操作,注册和登录如图 32、图 33所示。3.1.3. 协议服务器端和客户端之间互相通信需要使用的协议:l、all$msg为给所有人发送消息。2、直接输入view$获得在线用户列表。3、user$msg给用户名为user的用户发送私聊消息。4、trans$user$filenam将文件传输给who。5、reg为注册。6、login为登录。3.2. 聊天模块3.2.1. 客户端客户端在登陆成功之后在发送消息之前可以先查看在线用户列表,查看用户列表使用view$,如所示。图 36接着,可以使用user$msg的形式发送信息,这

11、个消息是发送给user的。或者,先使用user$来切换到发送消息,这个时候,你不需要加上协议,即可给user这个用户发送消息,如图 37、图 38所示。图 37图 38当然,上述方法也可以实现一对多聊天。如所示。图 39这些消息都加上协议user$msg来封装成结构体,再发送给服务器端。3.2.2. 服务器端如果是私聊,则根据客户端要发送到哪个用户,到链表里取得该用户名的客户端信息,服务器再发送给相应的接收信息的客户端,接收信息的客户终端就会光将信息保存到聊天记录的文件里,并显示收到的信息,并且信息前面会显示相应的提示符。3.3. 文件传输模块3.3.1. 客户端如果某个客户端想发送文件给其他

12、客户端,则直接使用命令trans$user$filename。trans为协议,user为用户名,filename包括本地的路径和文件名。如图 310、图 311所示。图 310图 3113.3.2. 服务器端当发送到服务器的时候,根据协议,先给接收的客户端发送一条消息为trans标记:某某用户给您发送了XX文件,Y接收N拒绝接收。如果客户端按下N,则返回一条消息给服务器端,服务器端同时告诉发送的客户端对方拒绝接收文件。发送端可以继续做其他的事。如果接收的客户端按下Y,并立即按受保存的路径和文件名。输入完后就可以接收文件了。客户端返回一条消息给无误器端,服务器端同时返回一条消息给发送的客户端,

13、告诉客户端对方同意接收文件,这时候文件开始传输;发送完毕后,发送的客户端会显示传输完毕。传输的文件可以使任意的文们。不仅仅是文本,多媒体文件也可以。第4章 实现本设计由C语言实现,用VIM编译。/*check.h*/#include #include #include #include #include #include #include #include #include #include #define MAXLEN 1024struct message char flag15; char name10; int size; char msgMAXLEN;int reg_check(str

14、uct message *recievemsg);int login_check(struct message *recievemsg);/*check.c*/#include check.hint reg_check(struct message *recievemsg) int fd; int read_size,write_size; struct message cmpmsg; if(strlen(recievemsg-name)10 |strlen(recievemsg-msg)20 ) return 1; if(strcmp(recievemsg-name,all)=0) retu

15、rn -1; if(strcmp(recievemsg-name,reg)=0) return -1; if(strcmp(recievemsg-name,login)=0) return -1; if(strcmp(recievemsg-name,trans)=0) return -1; if(fd=open(user.txt,O_RDWR|O_CREAT|O_A,PPEND,0666)0) perror(open); printf(openn); return -2; do if(read_size=read(fd,&cmpmsg,sizeof(cmpmsg)name,cmpmsg.nam

16、e)=0) close(fd); return -1; while(read_size = sizeof(struct message); if(write_size=write(fd,recievemsg,sizeof(struct message)0) perror(write); close(fd); return -2; while(write_size!=sizeof(struct message) /write_size = 0-writesize; lseek(fd,-write_size,SEEK_CUR); write_size=write(fd,recievemsg,siz

17、eof(struct message); printf(write file successn); close(fd); return 0;int login_check(struct message *recievemsg) int fd; struct message cmpmsg; int read_size; if(fd=open(user.txt,O_RDONLY)0) perror(open); return -2; do if(read_size=read(fd,&cmpmsg,sizeof(struct message)name,cmpmsg.name)=0)&(strcmp(

18、recievemsg-msg,cmpmsg.msg)=0) close(fd); return 0; while(read_size0); close(fd); return -1;/*void main()struct message sendmsg;printf(input name:n);gets(sendmsg.name);printf(input mima:n);gets(sendmsg.msg); printf(%dn,reg_check(&sendmsg); /printf(%dn,login_check(&sendmsg); */*client.c*/#include #inc

19、lude #include #include #include #include #include #include #include #include #include #include #include #include #define MAXLEN 1024struct message char flag15; char name10; int size; char msgMAXLEN;struct msq long msg_type; char msg_text5;int qid = -1,fd = -1,sockfd,savefilefd=-1;char filefromname10

20、;void handleQuit(int signal_no) if(fd 0) close(fd); close(sockfd); if(qid 0) if(msgctl(qid,IPC_RMID,NULL)0) printf(消息队列无法关闭n); exit(1); close(savefilefd); printf(程序正常退出n); raise(SIGQUIT);void cutStr(char str,char left, int n, char right,int m, char c) int i,k,j; for(i = 0 ; i n ;i+) if(stri = c) bre

21、ak; if(i = n) i = -1; else memset(left,0,strlen(left); for(k = 0 ; k i ; k+) leftk = strk; for(j = i+1 ; j m;j+) if(strj = 0) break; rightj-i-1 = strj; lefti = 0; if(j 0) printf(filedata.msg= %sn,filedata.msg); send(sockfd,&filedata,sizeof(structmessage),0); else printf(读取文件失败,文件传输中止n); break; while

22、 (filedata.size 0); close(savefilefd); savefilefd = -1 ;void handlerecvmsg(int *sockfd) int connfd = *sockfd; int nread; char buf1024; char str1024; struct message recvmsg; time_t timep; struct msq msg; if( fd =open(chatlog.txt,O_RDWR|O_CREAT|O_APPEND) 0) printf(打开聊天记录文件失败!); exit(1); / printf(%dn,f

23、d); if(qid = msgget(2222,IPC_CREAT|0666) = -1) printf(创建消息队列失败n); exit(1); msg.msg_type = getpid(); strcpy(msg.msg_text,OK); while(1) nread = recv(connfd,&recvmsg,sizeof(struct message),0); if(nread = 0) printf(与服务器断开了连接n); close(fd); close(connfd); exit(0); else if (strcmp(recvmsg.flag,all) = 0) ti

24、me (&timep); sprintf(str,%s%s发给所有人:%snn,ctime(&timep),recvmsg.name,recvmsg.msg); else if (strcmp(recvmsg.flag,sermsg) = 0) time (&timep); printf(%s服务器发给所有人:%snn,ctime(&timep),recvmsg.msg); continue; else if (strcmp(recvmsg.flag,view) = 0) time (&timep); printf(%s当前在线客户端:n%snn,ctime(&timep),recvmsg.m

25、sg); continue; else if (strcmp(recvmsg.flag,trans) = 0) pthread_t pid; if (strcmp(recvmsg.msg,agree) = 0) strcpy(filefromname,recvmsg.name); /创建线程发送文件 pthread_create(&pid,NULL,(void *)handlesendfile,NULL); else if(strcmp(recvmsg.msg,disagree) = 0) printf(对方拒绝接收文件n); close(savefilefd); savefilefd = -

26、1; else if(strcmp(recvmsg.msg,noexist) = 0) printf(该客户端不存在n); close(savefilefd); savefilefd = -1; else strcpy(filefromname,recvmsg.name); printf(%s向你请求传名为%s文件,是否同意接受?agree(同意)|disagree(不同意)n,recvmsg.name,recvmsg.msg); savefilefd = 0; continue; else if(strcmp(recvmsg.flag,transf) = 0) int n; if(strcm

27、p(recvmsg.msg,end$) = 0) printf(文件传输结束n); close(savefilefd); savefilefd = -1; continue; else n=write(savefilefd,recvmsg.msg,recvmsg.size); / printf(recvmsg.msg = %sn,recvmsg.msg); while(n 0) lseek(savefilefd,n,SEEK_CUR); n=write(savefilefd,recvmsg.msg,recvmsg.size); continue; else time (&timep); spr

28、intf(str,%s%s发来的私聊消息:%snn,ctime(&timep),recvmsg.name,recvmsg.msg); write(fd,str,strlen(str); msgsnd(qid,&msg,sizeof(struct msq),0); int main(int argc,char *argv) struct sockaddr_in server_addr; int port; int do_number; struct message a; char strMAXLEN; char bufMAXLEN; pthread_t pid; if(argc != 3) pr

29、intf(请输入服务器IP和端口n); exit(1); port = atoi(argv2); if(sockfd =socket(AF_INET,SOCK_STREAM,0) = -1) printf(创建socket失败n); exit(1); signal(SIGINT,handleQuit); printf(-n); printf(|n); printf(| input a number to work |n); printf(|t1.loginttt |n); printf(|t2.registertt |n); printf(|t3.exitttt |n); printf(|n)

30、; printf(-n); scanf(%d,&do_number); gets(str); while(do_number != 1 & do_number != 2 &do_number != 3) printf(你输入的不是上面的选项,请重新输入:n); scanf(%d,&do_number); gets(str); if(do_number=3) close(sockfd); printf(程序已退出!n); exit(0); bzero(&server_addr,sizeof(struct sockaddr_in); server_addr.sin_family = AF_INET

31、; server_addr.sin_addr.s_addr = inet_addr(argv1); server_addr.sin_port = htons(port); if(connect(sockfd,(struct sockaddr*)&server_addr,sizeof(struct sockaddr) = -1) printf(与服务器无响应,请隔一段时间再连接n); exit(2); if(do_number =1) int n = 3; while(n) printf(请输入你的用户名:n); scanf(%s,a.name); printf(请输入密码:n); scanf(

32、%s,a.msg); strcpy(a.flag,login); /a.flag3 = 0; send(sockfd,&a,sizeof(a),0); printf(正在等待服务器应答.n); recv(sockfd,buf,MAXLEN,0); printf(接到服务器发来的信息:%sn,buf); if(strcmp(buf,登录成功!) = 0) /int i,j,k; pthread_create(&pid,NULL,(void*)handlerecvmsg,(void *)&sockfd); gets(str); strcpy(a.flag,all); while(1) memset

33、(a.msg,0,strlen(a.msg); memset(str,0,strlen(str); gets(str); strcpy(buf,a.flag); cutStr(str,a.flag,15,a.msg,MAXLEN,$); printf(标志信息为:%sn,a.flag); if(strcmp(a.flag,view) = 0) send(sockfd,&a,sizeof(a),0); strcpy(a.flag,buf); continue; else if (strcmp(a.flag,trans) = 0) & (savefilefd =0) / printf(f=%s,a

34、=%s,s=%d,a.flag,a.msg,savefilefd); if (strcmp(a.msg,agree) = 0) & (savefilefd = 0) charsavefilename20; /charsavefileallname22; printf(请输入保存的文件名,文件将保存在当前目录下!n); do gets(savefilename); savefilefd = open(savefilename,O_RDWR|O_CREAT|O_EXCL,0666); if(savefilefd = -1) printf(文件名可能存在请重新命名.n); while(savefil

35、efd = -1); if(savefilefd 0) printf(创建文件失败!n); savefilefd = -1; else strcpy(a.name,filefromname); send(sockfd,&a,sizeof(a),0); / printf(agree :%s,%s,%sn,a.flag,a.name,a.msg); else memset(a.name,0,strlen(a.name); memset(str,0,strlen(str); cutStr(a.msg,a.name,10,str,MAXLEN,$); if (str0 != 0 & a.name0 !

36、= 0) chartransfileallname22; sprintf(transfileallname,./%s,str); savefilefd = open(str,O_RDWR,0666); if(savefilefd 0) printf(打开文件失败!n); savefilefd = -1; else memset(a.msg,0,strlen(a.msg); strcpy(a.msg,str); send(sockfd,&a,sizeof(a),0); /printf(tansmit :%s,%s,%s,a.flag,a.name,a.msg); else strcpy(a.msg,disagree); strcpy(a.name,filefromname); send(sockfd,&a,sizeof(a),0); / printf(disagree :%s,%s,%s,a.flag,a.name,a.msg); strcpy(a.flag,buf); continue; if (strcmp(a.flag,trans) = 0) strcpy(a.fla

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 教育专区 > 小学资料

本站为文档C TO C交易模式,本站只提供存储空间、用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。本站仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知淘文阁网,我们立即给予删除!客服QQ:136780468 微信:18945177775 电话:18904686070

工信部备案号:黑ICP备15003705号© 2020-2023 www.taowenge.com 淘文阁