linux系統下實現聊天室

目標:

 linux系統下實現聊天室linux

(1)在Linux系統下,使用TCP協議套接字編程;編程

(2)服務器應具備處理多個客戶端鏈接能力(設定最大鏈接數,如5個);服務器

(3)具備羣發和私聊的能力;數據結構

(4)過程描述socket

客戶端:鏈接服務器後,應能接收服務器發來信息並處理的能力,當收到聊天信息時,顯示羣發或私聊、信息發送客戶及發送的信息,當收到客戶加入或退出時,顯示客戶登陸或退出,並更新客戶列表;每次可選擇羣發或私聊,羣發時將鍵盤輸入信息發送給服務器,私聊時,選擇私聊客戶並將輸入信息發送給服務器。選擇退出時,結束進程或線程,關閉程序。this

服務器:爲每一個客戶鏈接建立一個進程或線程,處理客戶信息,當有新客戶加入或有客戶退出時,將客戶加入或退出信息發送給每一個客戶端;當收到某客戶的羣發信息時,將信息轉發給每一個客戶,當收到客戶私聊時將信息轉發給私聊客戶;客戶退出時關閉相應的進程或線程。編碼



2、系統設計

一、數據結構設計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.客戶端:

 

 

 

 

 

 

 

 

 

 

 

3、編碼實現

   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

相關文章
相關標籤/搜索