Linux下类似QQ聊天工具.docx

上传人:小飞机 文档编号:3160647 上传时间:2023-03-11 格式:DOCX 页数:36 大小:48.08KB
返回 下载 相关 举报
Linux下类似QQ聊天工具.docx_第1页
第1页 / 共36页
Linux下类似QQ聊天工具.docx_第2页
第2页 / 共36页
Linux下类似QQ聊天工具.docx_第3页
第3页 / 共36页
Linux下类似QQ聊天工具.docx_第4页
第4页 / 共36页
Linux下类似QQ聊天工具.docx_第5页
第5页 / 共36页
亲,该文档总共36页,到这儿已超出免费预览范围,如果喜欢就下载吧!
资源描述

《Linux下类似QQ聊天工具.docx》由会员分享,可在线阅读,更多相关《Linux下类似QQ聊天工具.docx(36页珍藏版)》请在三一办公上搜索。

1、Linux下类似QQ聊天工具概述 Abstract Linux作为免费开源操作系统广泛应用于企业。Linux与UNIX有着深厚的渊源,而UNIX系统可谓“坚如磐石”,其稳定性受到广泛赞誉。近年来,越来越多的就业岗位被提供给了Linux开发人员。 IM即时通讯软件是近年来流行的通信方式,企业、个人等都在更多地使用IM进行沟通。通过计算机网络,信息的传递变得十分方便。并非所有IM软件都要想腾讯QQ那样复杂、庞大,作为内部沟通,只需功能齐全、操作方便即可。 作为课程设计作品,在Linux下开发C/S式IM软件,对于熟悉Linux、复习计算机网络、进一步学习软件开发都有十分积极的意义。 As is o

2、pen 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 UNIX. IM software has become the most widely used communication software both for business use and for personal use. Throu

3、gh 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 Tencent QQ. IM software inside an Enterprise LAN should be lighter and easier. As a product design of Linux, developin

4、g 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. 内容摘要 随着嵌入式在生活中的应用越来越广泛,嵌入式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 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,等待连接,当有客户端连接服务器的时候,服务器接收连接,并接受客户端发送过来的消息,恨据接收到的结构

7、体所携带的协议来做相应的功能:服务器端启动后如图 31所示。 图 31 l、注册:如果协议为reg,则为客户端注册,首先将发送过来的结构体,提取用户名和密码,然后需要对川户名合法性检验,验证之后如果用户名合法则将川户信已保存到文件中,合法性的规则包括川户名不能重复和不能使用all等协议作为川户名,并且川户名和密码都不能为空。如果注册成功,服务器端发送一个消息给注册的客户端,同样将消息保存在一个结构休里如果失败,也给客户端发送一个消息如“您榆入的用户名小能为akk”或者“用户名XX已经存在”。注册结果如图 32所示。 图 32 2、登录:如果协议为login,则将用户名和密码信息提取,再遍历存放

8、用户信息文件里的用户名和密码直到验证成功为止,如果验证成功则对所有在线的川户发送一条消息:“提示XX用户登录成功”;如果失败则只给登阳失败的客户端提示登录失败,并给出原囚,如“用户名不存在”或者“用户名或者密码输入错误”,并跳转到相应的代码执行其他功能,成功则等特发送客户端消息,失败则关闭Socket并结束线程,如图 33所示: 图 33 3、监听和踢出客户端:通过六行和修改绑定的Socket和在线用户队列实现查看和踢出在线用户,踢出用户后向被踢出用户发送相关信息,如图 34、图 35所示。 图 34 图 35 3.1.2. 客户端 客户端的输入和消息的显示要使用2个终端,一个是Client,

9、一个是Display。Client终端为输入的界面。在这个界面里,新建一个线程来接受服务器端发来的消息,再添加时间信息,并将这些信息写入文件,然后给Display进程发送一个消息,Display进程接到消息,就去读取文件并将这些数拟显示在Display终端。 打开客户端Display终端界而,用lseek将内部指t阵行向文件末尾,等待Client终端里的线秤将消息写入文件。一旦有消息过来就去文件里读取数据井打印在Display终端。 打开客户端Client终端界而,有3个菜单,包括注册、登录、退出。选择相应项即可进幻相关操作,注册和登录如图 32、图 33所示。 3.1.3. 协议 服务器端和

10、客户端之间互相通信需要使用的协议: 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的形式发送信息,这个消息是发送给user的。或者,先使用user$来切换到发送消息,这个时候,你不需要加上协议,即可给user这个用户发送

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

12、ilename包括本地的路径和文件名。如图 310、图 311所示。 图 310 图 311 3.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 1024 struct message char flag15; char name10; int size; char msgMAXLEN; ; int reg_check(struct message *recievemsg); int login_ch

14、eck(struct message *recievemsg); /*check.c*/ #include check.h int 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,

15、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.name)=0) close(fd); return -1; while

16、(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,sizeof(struct message); printf(write

17、 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(recievemsg-msg,cmpmsg.msg)=0) c

18、lose(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 #include #include #include

19、 #include #include #include #include #include #include #include #include #include #include #define MAXLEN 1024 struct 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 handleQ

20、uit(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) break; if(i = n

21、) 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(struct message),0); else printf(读取文件失败,文件传输中止n); break; while (filedata

22、.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,fd); if(q

23、id = 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) time (&timep

24、); sprintf(str,%s%s发给所有 人:%snn,ctime(&timep),recvmsg.name,recvmsg.msg); else if else if (strcmp(recvmsg.flag,trans) = 0) (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当前在线

25、客户端:n%snn,ctime(&timep),recvmsg.msg); continue; 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 = -1; else

26、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(strcmp(recvms

27、g.msg,end$) = 0) printf(文件传输结束n); close(savefilefd); savefilefd = -1; continue; else n=write(savefilefd,recvmsg.msg,recvmsg.size); / printf(recvmsg.msg vmsg.name,recvmsg.msg); write(fd,str,strlen(str); msgsnd(qid,&msg,sizeof(struc= %sn,recvmsg.msg); while(n 0) lseek(savefilefd,n,SEEK_CUR); n=write(s

28、avefilefd,recvmsg.msg,recvmsg.size); continue; else time (&timep); sprintf(str,%s%s发来的私聊消息:%snn,ctime(&timep),rect 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(请输入

29、服务器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); pri

30、ntf(-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; s

31、erver_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) memse

33、t(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,

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

35、vefilefd = -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.na

36、me0 != 0) char transfileallname22; 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 strcp

37、y(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.flag,buf); send(sockfd,&a,sizeof(a),0); else n-; printf(您还有%d次机会,之后将推出程序!n,n); close(sockfd); exit(3); /登陆 else if(do_number =2) /register int i =1 ; char use

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 生活休闲 > 在线阅读


备案号:宁ICP备20000045号-2

经营许可证:宁B2-20210002

宁公网安备 64010402000987号