《Linux下的简单聊天工具(共24页).doc》由会员分享,可在线阅读,更多相关《Linux下的简单聊天工具(共24页).doc(24页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、精选优质文档-倾情为你奉上概述AbstractLinux作为免费开源操作系统广泛应用于企业。Linux与UNIX有着深厚的渊源,而UNIX系统可谓“坚如磐石”,其稳定性受到广泛赞誉。近年来,越来越多的就业岗位被提供给了Linux开发人员。IM即时通讯软件是近年来流行的通信方式,企业、个人等都在更多地使用IM进行沟通。通过计算机网络,信息的传递变得十分方便。并非所有IM软件都要想腾讯QQ那样复杂、庞大,作为内部沟通,只需功能齐全、操作方便即可。作为课程设计作品,在Linux下开发C/S式IM软件,对于熟悉Linux、复习计算机网络、进一步学习软件开发都有十分积极的意义。As is open so
2、urce 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 UNIX. IM software has become the most widely used communication software both for business use and for personal use. Through int
3、ernetworks - 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 Tencent QQ. IM software inside an Enterprise LAN should be lighter and easier. As a product design of Linux, developing an I
4、M software is helpful to reviewing key points of Linux, Internet, and software developing. 专心-专注-专业目 录第1章 设计任务1.1. 设计任务概述1.1.1. 课题名称Linux下的IM网络聊天软件1.1.2. 内容摘要随着嵌入式在生活中的应用越来越广泛,嵌入式LINUX下的Socket网络编程也越来越热。为了加强对在Linux系统下进行编程的学习和实践,运用计算机网络、软件工程等知识,我们选择了Linux下网络聊天工具作为课设题目。本系统主要功能在于实现多个客户端一与服务器端之问的信息传递与文件传
5、输功能。系统主要实现4人聊天室功能:注册与登录系统,公聊,私聊,文件传输。应用LINUX下的Socket网络编程并使用TCP实现简单聊天程序。编程过程中不断用各种网络调试工具进行调试程序,最终做成了简单的在Linux下的聊天工具。关键词:Linux Socket IM软件1.2. 开发环境1.2.1. 硬件环境Intel x86 CPU架构(主频1G Hz以上)512MB DDR2内存支持800*600以上分辨率、16色以上的显示设备1.2.2. 软件环境Red Hat Enterprise Linux(或Fedora)VIM编译器1.3. 要求在Linux下实现网络聊天,包括公聊、一对多私聊
6、等功能。实现客户端之间经网络传输文件。能保存聊天记录。第2章 设计方案2.1. 系统结构2.1.1. 结构结构组成:注册、登录、聊天、文件传输。2.1.2. 结构图聊天系统分为五大模块2.2. 流程与模块2.2.1. 执行流程2.2.2. 各功能需要调用的模块模块需求注册登录聊天文件传输用户注册用户登录系统广播上下线提示用户私聊文件传输强制下线第3章 实现原理3.1. 注册、登录模块3.1.1. 服务器端服务器端建立好socket,等待连接,当有客户端连接服务器的时候,服务器接收连接,并接受客户端发送过来的消息,恨据接收到的结构体所携带的协议来做相应的功能:服务器端启动后如Error! Ref
7、erence source not found.所示。图 31l、注册:如果协议为reg,则为客户端注册,首先将发送过来的结构体,提取用户名和密码,然后需要对川户名合法性检验,验证之后如果用户名合法则将川户信已保存到文件中,合法性的规则包括川户名不能重复和不能使用all等协议作为川户名,并且川户名和密码都不能为空。如果注册成功,服务器端发送一个消息给注册的客户端,同样将消息保存在一个结构休里如果失败,也给客户端发送一个消息如“您榆入的用户名小能为akk”或者“用户名XX已经存在”。注册结果如Error! Reference source not found.所示。图 322、登录:如果协议为l
8、ogin,则将用户名和密码信息提取,再遍历存放用户信息文件里的用户名和密码直到验证成功为止,如果验证成功则对所有在线的川户发送一条消息:“提示XX用户登录成功”;如果失败则只给登阳失败的客户端提示登录失败,并给出原囚,如“用户名不存在”或者“用户名或者密码输入错误”,并跳转到相应的代码执行其他功能,成功则等特发送客户端消息,失败则关闭Socket并结束线程,如Error! Reference source not found.所示:图 333、监听和踢出客户端:通过六行和修改绑定的Socket和在线用户队列实现查看和踢出在线用户,踢出用户后向被踢出用户发送相关信息,如Error! Refere
9、nce source not found.、Error! Reference source not found.所示。图 34图 353.1.2. 客户端客户端的输入和消息的显示要使用2个终端,一个是Client,一个是Display。Client终端为输入的界面。在这个界面里,新建一个线程来接受服务器端发来的消息,再添加时间信息,并将这些信息写入文件,然后给Display进程发送一个消息,Display进程接到消息,就去读取文件并将这些数拟显示在Display终端。打开客户端Display终端界而,用lseek将内部指t阵行向文件末尾,等待Client终端里的线秤将消息写入文件。一旦有消息过
10、来就去文件里读取数据井打印在Display终端。打开客户端Client终端界而,有3个菜单,包括注册、登录、退出。选择相应项即可进幻相关操作,注册和登录如Error! Reference source not found.、Error! Reference source not found.所示。3.1.3. 协议服务器端和客户端之间互相通信需要使用的协议:l、all$msg为给所有人发送消息。2、直接输入view$获得在线用户列表。3、user$msg给用户名为user的用户发送私聊消息。4、trans$user$filenam将文件传输给who。5、reg为注册。6、login为登录。3.
11、2. 聊天模块3.2.1. 客户端客户端在登陆成功之后在发送消息之前可以先查看在线用户列表,查看用户列表使用view$,如所示。图 36接着,可以使用user$msg的形式发送信息,这个消息是发送给user的。或者,先使用user$来切换到发送消息,这个时候,你不需要加上协议,即可给user这个用户发送消息,如Error! Reference source not found.、Error! Reference source not found.所示。图 37图 38当然,上述方法也可以实现一对多聊天。如所示。图 39这些消息都加上协议user$msg来封装成结构体,再发送给服务器端。3.2.
12、2. 服务器端如果是私聊,则根据客户端要发送到哪个用户,到链表里取得该用户名的客户端信息,服务器再发送给相应的接收信息的客户端,接收信息的客户终端就会光将信息保存到聊天记录的文件里,并显示收到的信息,并且信息前面会显示相应的提示符。3.3. 文件传输模块3.3.1. 客户端如果某个客户端想发送文件给其他客户端,则直接使用命令trans$user$filename。trans为协议,user为用户名,filename包括本地的路径和文件名。如Error! Reference source not found.、Error! Reference source not found.所示。图 310图
13、 3113.3.2. 服务器端当发送到服务器的时候,根据协议,先给接收的客户端发送一条消息为trans标记:某某用户给您发送了XX文件,Y接收N拒绝接收。如果客户端按下N,则返回一条消息给服务器端,服务器端同时告诉发送的客户端对方拒绝接收文件。发送端可以继续做其他的事。如果接收的客户端按下Y,并立即按受保存的路径和文件名。输入完后就可以接收文件了。客户端返回一条消息给无误器端,服务器端同时返回一条消息给发送的客户端,告诉客户端对方同意接收文件,这时候文件开始传输;发送完毕后,发送的客户端会显示传输完毕。传输的文件可以使任意的文们。不仅仅是文本,多媒体文件也可以。第4章 实现本设计由C语言实现,
14、用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(struct message *recievemsg);int login_check(struct message *recievemsg);/*check.c*/#include c
15、heck.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) return -1; if(strcmp(recievemsg-name,reg)=0) return -1; if(strcmp(recievemsg-name,login)=0) re
16、turn -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.name)=0) close(fd); return -1; while(read_size = sizeof(struct message); if(write_size=write(
17、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,sizeof(struct message); printf(write file successn); close(fd); return 0;int login_check(stru
18、ct 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(recievemsg-msg,cmpmsg.msg)=0) close(fd); return 0; while(read_size0); close(fd); return -1
19、;/*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 #include #include #include #include #include #include #include #include #include #include #inc
20、lude #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;void handleQuit(int signal_no) if(fd 0) close(fd); close(sockfd); if(qid 0) if(msgctl(qid
21、,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) break; if(i = n) i = -1; else memset(left,0,strlen(left); for(k = 0 ; k i ; k+) leftk = strk;
22、 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 (filedata.size 0); close(savefilefd); savefilefd = -1 ;void handlerecvmsg(int *sockfd) in
23、t 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,fd); if(qid = msgget(2222,IPC_CREAT|0666) = -1) printf(创建消息队列失败n); exit(1); msg.msg_type =
24、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) time (&timep); sprintf(str,%s%s发给所有人:%snn,ctime(&timep),recvmsg.name,recvmsg.msg); else if (
25、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.msg); continue; else if (strcmp(recvmsg.flag,trans) = 0) pthread_t pid; if (strcmp(recvmsg.
26、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 = -1; else if(strcmp(recvmsg.msg,noexist) = 0) printf(该客户端不存在n); close(savefilefd); savefilef
27、d = -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(strcmp(recvmsg.msg,end$) = 0) printf(文件传输结束n); close(savefilefd); savefilefd = -1; continue; el
28、se 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); sprintf(str,%s%s发来的私聊消息:%snn,ctime(&timep),recvmsg.name,recvmsg.msg); write(fd,str,strlen(str
29、); 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) printf(请输入服务器IP和端口n); exit(1); port = atoi(argv2); if(sockfd =socket(AF_INET,SOCK_STREAM,0)
30、= -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); printf(-n); scanf(%d,&do_number); gets(str); while(do_number != 1 & do_number != 2 &do_n
31、umber != 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; server_addr.sin_addr.s_addr = inet_addr(argv1); server_addr.sin_port = htons(port); if(c
32、onnect(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(%s,a.msg); strcpy(a.flag,login); /a.flag3 = 0; send(sockfd,&a,sizeof(a),0); printf(正在等待服务器
33、应答.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(a.msg,0,strlen(a.msg); memset(str,0,strlen(str); gets(str); strcpy(buf,a.flag); cutStr(st
34、r,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=%s,s=%d,a.flag,a.msg,savefilefd); if (strcmp(a.msg,agree) = 0) & (savefilefd = 0) charsav
35、efilename20; /charsavefileallname22; printf(请输入保存的文件名,文件将保存在当前目录下!n); do gets(savefilename); savefilefd = open(savefilename,O_RDWR|O_CREAT|O_EXCL,0666); if(savefilefd = -1) printf(文件名可能存在请重新命名.n); while(savefilefd = -1); if(savefilefd 0) printf(创建文件失败!n); savefilefd = -1; else strcpy(a.name,filefrom
36、name); 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 != 0) chartransfileallname22; sprintf(transfileallname,./%s,str); savefilefd = open(str,O_R
37、DWR,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;