linux系統下實現聊天室linux
(1)在Linux系統下,使用TCP協議套接字編程;編程
(2)服務器應具備處理多個客戶端鏈接能力(設定最大鏈接數,如5個);服務器
(3)具備羣發和私聊的能力;數據結構
(4)過程描述socket
客戶端:鏈接服務器後,應能接收服務器發來信息並處理的能力,當收到聊天信息時,顯示羣發或私聊、信息發送客戶及發送的信息,當收到客戶加入或退出時,顯示客戶登陸或退出,並更新客戶列表;每次可選擇羣發或私聊,羣發時將鍵盤輸入信息發送給服務器,私聊時,選擇私聊客戶並將輸入信息發送給服務器。選擇退出時,結束進程或線程,關閉程序。this
服務器:爲每一個客戶鏈接建立一個進程或線程,處理客戶信息,當有新客戶加入或有客戶退出時,將客戶加入或退出信息發送給每一個客戶端;當收到某客戶的羣發信息時,將信息轉發給每一個客戶,當收到客戶私聊時將信息轉發給私聊客戶;客戶退出時關閉相應的進程或線程。編碼
一、數據結構設計spa
A.定義在線客戶列表數據結構ClientList。.net
typedef struct ClientList 線程
{
char name[NAME_LEN];
int socketFd;
}CLIENTLIST;
說明:
name爲客戶端名稱;socketFd是服務器爲客戶端分配的通訊 socket套接字標識符;
B.定義客戶與服務器之間交互信息格式結構Message。
typedef struct Message
{
char fromUser[NAME_LEN];
int fromUserLocate;
int type;
int sendUserLocate;
char message[MSG_LEN];
CLIENTLIST clientList[MAX_CLIENT];
}MESSAGE;
說明:
fromUser爲客戶端名稱;fromUserLocate爲該客戶端在在線客戶列表 裏的位置;type爲消息的類型(包括羣發、私聊、登陸、退出); sendUserLocate爲私聊對象在客戶端列表中的位置(僅在type爲私 聊類型是有效);message爲發送的消息內容,clientList爲在線客戶
列表。
C.定義在服務器端主線程爲接收線程傳遞的數據結構ARG。
struct ARG
{
int locate;
int fifoFd;
};
說明:
locate爲與之通訊的客戶端在在線客戶列表中的位置;
fifoFd爲以只寫方式打開的管道標識符;
D.客戶端中的全局變量
pthread_t tid1; //接收線程的標識符
char g_name[NAME_LEN];//客戶端用戶的名稱
int g_locate; //客戶端在在線客戶列表中的位置
int g_total; //在線客戶的總數
二、流程設計
A. 服務器端:
B.客戶端:
Public.h
#ifndef PUBLIC_H_
#define PUBLIC_H_
#include"stdio.h"
#include"stdlib.h"
#include"string.h"
#include"sys/socket.h"
#include"sys/types.h"
#include"arpa/inet.h"
#include"netinet/in.h"
#include"unistd.h"
#include"pthread.h"
#define MAX_CLIENT 3
#define NAME_LEN 20
#define MSG_LEN 100
#define PORT 12345
#define LOGIN 1
#define EXIT 2
#define PUBLIC 3
#define PRIVATE 4
#define OK 5
#define ERROR -6
typedef struct ClientList
{
char name[NAME_LEN];
int socketFd;
}CLIENTLIST;
typedef struct Message
{
char fromUser[NAME_LEN];
int fromUserLocate;
int type;
int sendUserLocate;
char message[MSG_LEN];
CLIENTLIST clientList[MAX_CLIENT];
}MESSAGE;
CLIENTLIST clientList[MAX_CLIENT];
#endif
Server.c
#include"public.h"
#include"sys/stat.h"
#include"fcntl.h"
#define ADD 7
#define DEL 8
#define FIFO "FIFO"
struct ARG
{
int locate;
int fifoFd;
};
int SearchLocate()
{
int i;
for(i=0;i<MAX_CLIENT;i++)
{
if(clientList[i].socketFd==0)break;
}
if(i<MAX_CLIENT) return i; else return -1;
}
void TransmitMsg(int cmd,int locate,MESSAGE msg)
{
memcpy(&msg.clientList,&clientList,sizeof(clientList));
if(cmd==PRIVATE)
{
write(clientList[msg.sendUserLocate].socketFd,&msg,
sizeof(msg));
printf("\e[31m#PRIVATE > From:%-5s To: %-5s
Msg:%s\e[0m\n",clientList[locate].name,clientList[msg.se ndUserLocate].name,msg.message);
}
else
{
int i;
for (i=0;i<MAX_CLIENT;i++)
{
if(clientList[i].socketFd!=0 && i!=locate)
{
write(clientList[i].socketFd,&msg,sizeof(msg));
printf("\e[32m#PUBLIC > From:%-5s To: %-5s
Msg:%s\e[0m\n",clientList[locate].name,
clientList[i].name,msg.message);
}
}
if(cmd==LOGIN)
{
write(clientList[locate].socketFd,&msg,sizeof(msg));
}
}
}
void UpdateList(int cmd , char *name,int locate)
{
if(cmd==ADD)
{
strcpy(clientList[locate].name,name);
printf("\e[33m*ADD USER> NAME:%-5s
\e[0m\n",clientList[locate].name);
}
else if(cmd==DEL)
{
printf("\e[33m*DEl USER> NAME:%-5s \e[0m\n",clientList[locate].name);
clientList[locate].socketFd=0;
bzero(clientList[locate].name,NAME_LEN);
}
}
void *RecvMsg(void *arg_t)
{
struct ARG arg=*(struct ARG *)arg_t;
MESSAGE msg;
while(1)
{
int flag;
bzero(&msg,sizeof(msg)); msg.type=ERROR;
read(clientList[arg.locate].socketFd,&msg,sizeof(msg));
msg.fromUserLocate=arg.locate;
if(msg.type==EXIT||msg.type==ERROR)
{
if(msg.type==ERROR)
{
strcpy(msg.message,"breakdown");
printf("\e[33m*CLIENT:%s HAD BREAKDOWN
\e[0m\n",clientList[msg.fromUserLocate].name);
msg.type=EXIT;
}
if(-1==(flag=write(arg.fifoFd,&msg,sizeof(msg))))
{
perror("write fifo error");
exit(1);
}
break;
}
if(-1==(flag=write(arg.fifoFd,&msg,sizeof(msg))))
{
perror("write fifo error");
exit(1);
}
}
return NULL;
}
void *SendMsg(void *fd)
{
int fifoFd;
if(-1==(fifoFd=open(FIFO,O_RDONLY)))
{
perror("open fifo error");
exit(1);
}
int flag;
MESSAGE msg;
while(1)
{
if(-1==(flag=read(fifoFd,&msg,sizeof(msg))))
{
perror("read fifo error");
exit(2);
}
int exit_fd;
switch(msg.type)
{
case LOGIN:
UpdateList(ADD,msg.fromUser,msg.fromUserLocate);
TransmitMsg(LOGIN,msg.fromUserLocate,msg);
break;
case PUBLIC:
TransmitMsg(PUBLIC,msg.fromUserLocate,msg);
break;
case PRIVATE:
TransmitMsg(PRIVATE,msg.fromUserLocate,msg);
break;
case EXIT:
exit_fd=clientList[msg.fromUserLocate].socketFd;
UpdateList(DEL,msg.fromUser,msg.fromUserLocate);
TransmitMsg(EXIT,msg.fromUserLocate,msg);
close(exit_fd);
break;
default:
printf("bad data %d \n",msg.type);
break;
}
}
return NULL;
}
int main()
{
printf("\n\tservice is start.....\n");
pthread_t tid1,tid2;
int fd,clientfd,wr_fifo;
socklen_t sock_len;
sock_len=sizeof(struct sockaddr_in);
mkfifo(FIFO,O_CREAT|O_EXCL);
pthread_create(&tid1,NULL,SendMsg,NULL);
struct sockaddr_in server,client;
server.sin_port=htons(PORT);
server.sin_family=AF_INET;
server.sin_addr.s_addr=INADDR_ANY;
if(-1== (fd=socket(AF_INET,SOCK_STREAM,0)))
{
perror("socket error ");
exit(1);
}
if(-1==bind(fd,(struct sockaddr*)&server,sock_len))
{
perror("bind error");
exit(2);
}
if(-1==(listen(fd,MAX_CLIENT+1)))
{
perror("listen error");
exit(3);
}
if(-1==(wr_fifo=open(FIFO,O_WRONLY)))
{
perror("open fifo error");
exit(1);
}
while(1)
{
if(-1==(clientfd=(accept(fd,(struct
sockaddr*)&client,&sock_len))))
{
perror("accept error");
exit(4);
}
int locate=-1;
MESSAGE msg;
if(-1==(locate=SearchLocate()))
{
printf("\e[33m*RECEIVE A APPLY BUT CANNOT ALLOW
CONNECT\e[0m \n");
msg.type=EXIT;
write(clientfd,&msg,sizeof(msg));
}
else
{
struct ARG arg;
arg.fifoFd=wr_fifo;
arg.locate=locate;
msg.type=OK;
memcpy(&msg.clientList,&clientList,sizeof(clientList));
msg.fromUserLocate=locate;
write(clientfd,&msg,sizeof(msg));
clientList[locate].socketFd=clientfd;
pthread_create(&tid1,NULL,RecvMsg,(void *)&arg);
}
}
pthread_join(tid1,NULL);
pthread_join(tid2,NULL);
unlink("FIFO");
return 0;
}
Client.c
#include"public.h"
pthread_t tid1;
char g_name[NAME_LEN];
int g_locate;
int g_total;
void flush(){ char c; do{c=getc(stdin);}while(c!='\n'&&c!=EOF);};
int CheckExist()
{
int i;
for(i=0;i<MAX_CLIENT;i++)
{
if(!strcmp(g_name,clientList[i].name))
break;
}
if(i<MAX_CLIENT)
{
printf("this name: %s is already exist!!\n",g_name);
return 1;
}
else
return 0;
}
void ShowList()
{
int i;
g_total=0;
printf("\t _____________________________ \n");
printf("\t| CLIENT LIST |\n");
printf("\t|_____________________________|\n");
printf("\t|\e[4m sort | name \e[24m|\n ");
for(i=0;i<MAX_CLIENT;i++)
{
if(clientList[i].socketFd!=0)
{
if(i==g_locate)
{
printf("\t|\e[4;31m *%-4d | %-10s
\e[0m|\n",++g_total,clientList[i].name);
}
else
{
printf("\t|\e[4m %-4d | %-10s
\e[24m|\n",++g_total,clientList[i].name);
}
}
}
printf("\t|\e[4m Total:%-3d \e[24m|\n",g_total);
}
int MakeTempList(int *tmp)
{
int i,n=0;
for(i=0;i<MAX_CLIENT;i++)
{
if(clientList[i].socketFd!=0)
{ tmp[n]=i; n++; }
}
ShowList();
int select;
printf("please select the user \n");
if(1!=scanf("%d",&select))
{
flush();
printf("bad select \n");
return -1;
}
if(select<=g_total)
{
if(tmp[select-1]==g_locate)
{
printf("\e[33m#SYSTEM:YOU CAN NOT SELECT
YOURSELF\e[0m\n");
return -1;
}
else
return tmp[select-1];
}
else
{
printf("bad select \n");
return -1;
}
}
void *RecvMsg(void *fd)
{
int sockfd=*(int *)fd;
MESSAGE msg;
while(1)
{
bzero(&msg,sizeof(msg)); msg.type=ERROR;
read(sockfd,&msg,sizeof(msg));
if(msg.type==ERROR)
break;
switch(msg.type)
{
case LOGIN:
if(msg.fromUserLocate==g_locate)
printf("\e[34m###### > loing succeed\e[0m\n");
else
printf("\e[33m#LOGIN > From:%-10s
Msg:%s\e[0m\n",msg.fromUser,msg.message);
break;
case EXIT:
printf("\e[33m#EXIT > From:%-10s
Msg:%s\e[0m\n",clientList[msg.fromUserLocate].name,
msg.message);
break;
case PUBLIC:
printf("\e[32m#PUBLIC > From:%-10s
Msg:%s\e[0m\n",msg.fromUser,msg.message);
break;
case PRIVATE:
printf("\e[31m#PRIVATE> From:%-10s
Msg:%s\e[0m\n",msg.fromUser,msg.message);
break;
default:break;
}
memcpy(&clientList,&msg.clientList,sizeof(clientList));
}
printf("server is breakdown \n");
exit(1);
}
void SendMsg(int fd)
{
MESSAGE msg;
msg.type=LOGIN;
msg.fromUserLocate=g_locate;
strcpy(msg.fromUser,g_name);
strcpy(msg.message,g_name);
write(fd,&msg,sizeof(msg));
int tmp[MAX_CLIENT];
int key;
while(1)
{
printf(" 1 public 2 private 3 EXIT 4 client list\n");
if(1!= scanf("%d",&key))
{
key=0;
flush();
}
bzero(&msg,sizeof(msg));
strcpy(msg.fromUser,g_name);
msg.fromUserLocate=g_locate;
switch(key)
{
case 1:
msg.type=PUBLIC;
printf("\npublic: please input content \n");
flush();
fgets(msg.message,sizeof(msg.message),stdin);
msg.message[strlen(msg.message)-1]='\0';
write(fd,&msg,sizeof(msg));
break;
case 2:
bzero(tmp,sizeof(tmp));
msg.type=PRIVATE;
if(-1!=(msg.sendUserLocate=MakeTempList(tmp)))
{
printf("\nprivate: please input content \n");
flush();
fgets(msg.message,sizeof(msg.message),stdin);
msg.message[strlen(msg.message)-1]='\0';
write(fd,&msg,sizeof(msg));
}
break;
case 3:
printf("EXIT \n");
msg.type=EXIT;
strcpy(msg.message,"bye-bye");
write(fd,&msg,sizeof(msg));
break;
case 4:
ShowList();
break;
default:
printf("bad select \n");
msg.type=0;
break;
}
if(msg.type==EXIT)
{
break;
}
}
pthread_cancel(tid1);
}
int main()
{
int fd;
char ip[20]="127.0.0.1";
//printf("please input the ip \n");scanf("%s",ip);
struct sockaddr_in addr;
addr.sin_port=htons(PORT);
addr.sin_family=AF_INET;
addr.sin_addr.s_addr=inet_addr(ip);
if(-1==(fd=socket(AF_INET,SOCK_STREAM,0)))
{
perror("socket error");
exit(1);
}
if(-1==(connect(fd,(struct sockaddr*)&addr,sizeof(struct
sockaddr))))
{
perror("connect error");
exit(2);
}
MESSAGE msg;
read(fd,&msg,sizeof(msg));
if(msg.type==EXIT)
{
printf("service refuse connect \n");
exit(1);
}
else
{
memcpy(&clientList,&msg.clientList,sizeof(clientList));
g_locate=msg.fromUserLocate;
pthread_create(&tid1,NULL,RecvMsg,(void *)&fd);
do{
printf("please input your name\n");scanf("%s",g_name);
}while(CheckExist());
SendMsg(fd);
pthread_join(tid1,NULL);
}
return 0;
}
MAkeFile
tar:server client
server:server.c
gcc -g -Wall -o $@ $< -lpthread
client:client.c
gcc -g -Wall -o $@ $< -lpthread
c:
rm -rf server client FIFO
源代碼:http://download.csdn.net/detail/w1143408997/9408145