客户端
ubuntu@ubuntu:~/internet/exercise/04_多点通信$ cat chatroomCli.c
#include<stdio.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<sys/wait.h>
#include<signal.h>
#include<stdlib.h>
#define N 128
#define ERR_MSG(msg) do{\
printf("line:%d\n",__LINE__);\
perror(msg);\
}while(0)
#define PORT 1026 //端口号的网络字节序:1024~49151;
#define IP "192.168.10.110" //ifconfig查看到的本机IP
void zombie_handler(int sig)
{
while(waitpid(-1,NULL,WNOHANG)>0);
}
int main(int argc, const char *argv[])
{
//子进程退出,父未退,所以产生僵尸进程
if(signal(SIGCHLD,zombie_handler)==SIG_ERR)
{
ERR_MSG("signal");
return -1;
}
//创建流式套接字
int cfd = socket(AF_INET,SOCK_STREAM,0);
if(cfd<0)
{
ERR_MSG("socket");
return -1;
}
printf("socket create success,cfd=%d__%d__\n",cfd,__LINE__);
//功能:允许端口快速被重用,快速被复用
int reuse = 1;
if(setsockopt(cfd,SOL_SOCKET,SO_REUSEADDR,&reuse,sizeof(reuse))<0)
{
ERR_MSG("setsockopt");
return -1;
}
//填充服务器自身的地址信息结构体
//真实的地址信息结构体根据就地址族指定 AF_INET:man 7 ip
struct sockaddr_in sin;
sin.sin_family =AF_INET; //必须填充AF_INET
sin.sin_port =htons(PORT); //端口号的网络字节序,2字节的无符号整型数,16位:1~65535,其中我们自己用1024~49151范围内的
sin.sin_addr.s_addr=inet_addr(IP);//ifconfig查看本机IP
//登录
char ID[20]="";
printf("please input the ID:\n");
fgets(ID,sizeof(ID),stdin);
char L;
char buf[N]="";
int size = sprintf(buf,"%c%s",L,ID);
if(sendto(cfd,buf,size,0,(struct sockaddr*)&sin,sizeof(sin))<0)
{
ERR_MSG("sendto");
return -1;
}
pid_t cpid = fork();
if(cpid>0) //父进程接收系统消息,登录,群聊,下线
{
while(1)
{
bzero(buf,sizeof(buf));
//接收数据
ssize_t res = recv(cfd,buf,sizeof(buf),0);
if(res<0)
{
ERR_MSG("recv");
return -1;
}
printf("%s",buf); //客户端接收到来自服务器转发的信息:系统信息,登录、群聊、下线等
}
}
else if(0==cpid)
{
while(1)
{
//子进程从终端获取数据,quit时下线'Q',非quit群聊'C'
char arr[128]="";
bzero(arr,sizeof(arr));
fgets(arr,sizeof(arr),stdin);
if(0==strcasecmp(arr,"quit"))
{
bzero(arr,sizeof(arr));
strcat(arr,"下线");
sendto(cfd,arr,sizeof(arr),0,(struct sockaddr*)&sin,sizeof(sin)<0);
goto END;
}
else
{
sendto(cfd,arr,sizeof(arr),0,(struct sockaddr*)&sin,sizeof(sin)<0);
continue;
}
}
}
else
{
ERR_MSG("fork");
return -1;
}
END:
close(cfd);
return 0;
}
服务器
ubuntu@ubuntu:~/internet/exercise/04_多点通信$ cat chatroomSer.c
#include<stdio.h>
#include<string.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<stdlib.h>
#include<pthread.h>
#define ERR_MSG(msg) do{\
printf("line:%d\n",__LINE__);\
perror(msg);\
}while(0)
#define PORT 2333 //端口号的网络字节序:1024~49151;
#define IP "192.168.10.255" //广播IP
#define N 128
pthread_t tid;
//链表 传参
typedef struct snd
{
int sfd;
struct sockaddr_in cin;
struct sockaddr_in sin;
struct snd* next;
}*linklist;
//信息结构体
typedef struct
{
int type;
char name[30];
char text[N];
}MSG;
int do_login(int sfd,linklist L,MSG rcv_msg,struct sockaddr_in cin);
int do_chat(int sfd,linklist L,MSG rcv_msg,struct sockaddr_in cin);
int do_quit(int sfd,linklist L,MSG rcv_msg,struct sockaddr_in cin);
void* SndMsg(void* arg); //void* arg = (void*)&info
int main(int argc, const char *argv[])
{
if(argc<3)
{
printf("please input the ip&port\n");
return -1;
}
//创建报式套接字
int sfd = socket(AF_INET,SOCK_DGRAM,0);
if(sfd<0)
{
ERR_MSG("socket");
return -1;
}
printf("socket create success,sfd=%d__%d__\n",sfd,__LINE__);
//填充接收方的地址信息结构体
//真实的地址信息结构体根据就地址族指定 AF_INET:man 7 ip
struct sockaddr_in sin;
sin.sin_family =AF_INET; //必须填充AF_INET
sin.sin_port =htons(atoi(argv[2])); //端口号的网络字节序,2字节的无符号整型数,16位:1~65535,其中我们自己用1024~49151范围内的
sin.sin_addr.s_addr=inet_addr(argv[1]);//广播IP
//绑定接收方的IP和端口
if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))<0)
{
ERR_MSG("bind");
return -1;
}
printf("bind success __%d__\n",__LINE__);
//创建线程
struct snd info;
pthread_create(&tid,NULL,SndMsg,(void*)&info);
//创建链表存储客户端信息
linklist L= (linklist)malloc(sizeof(struct snd));
L->next = NULL;
//接收消息
char buf[128]="";
ssize_t res = 0;
struct sockaddr_in cin; //存储客户端的地址信息
socklen_t len = sizeof(cin);
MSG rcv_msg;
while(1)
{
bzero(rcv_msg.text,sizeof(rcv_msg.text));
//接收数据
res = recvfrom(sfd,&rcv_msg,sizeof(rcv_msg),0,(void*)&cin,&len);
if(res<0)
{
ERR_MSG("recvfrom");
return -1;
}
int type = ntohs(rcv_msg.type);
switch(type)
{
case 'L':
do_login(sfd,L,rcv_msg,cin);
break;
case 'C':
do_chat(sfd,L,rcv_msg,cin);
break;
case 'Q':
do_quit(sfd,L,rcv_msg,cin);
break;
default:
printf("input fasle\n");
}
}
if(close(sfd)<0)
{
ERR_MSG("close");
return -1;
}
return 0;
}
int do_login(int sfd,linklist L,MSG rcv_msg,struct sockaddr_in cin)
{
//登陆成功
printf("%s [%s:%d]登陆成功\n",rcv_msg.name,\
(char*)inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));
sprintf(rcv_msg.text,"------%s登陆成功-----",rcv_msg.name);
while(L->next!=NULL)
{
L=L->next;
if(sendto(sfd,&rcv_msg,sizeof(rcv_msg),0,(void*)&(L->cin),sizeof(L->cin))<0)
{
ERR_MSG("sendto");
return -1;
}
}
linklist temp = (linklist)malloc(sizeof(struct snd));
temp->cin = cin;
temp->next = NULL;
L->next = temp;
return 0;
}
int do_chat(int sfd,linklist L,MSG rcv_msg,struct sockaddr_in cin)
{
char buf[258]="";
sprintf(buf,"%s:%s",rcv_msg.name,rcv_msg.text);
strcat(rcv_msg.text,buf);
while(L->next!=NULL)
{
L = L->next;
if(memcmp(&cin,&L->cin,sizeof(cin))!=0)
{
if(sendto(sfd,&rcv_msg,sizeof(rcv_msg),0,(void*)&(L->cin),sizeof(L->cin))<0)
{
ERR_MSG("sendto");
return -1;
}
}
}
return 0;
}
int do_quit(int sfd,linklist L,MSG rcv_msg,struct sockaddr_in cin)
{
if(0==strncasecmp(rcv_msg.text,"quit",4))
{
char buf[258]="";
sprintf(buf,"%s:%s",rcv_msg.name,"用户下线");
strcat(rcv_msg.text,buf);
while(L->next!=NULL)
{
L = L->next;
if(memcmp(&cin,&L->cin,sizeof(cin))!=0)
{
if(sendto(sfd,&rcv_msg,sizeof(rcv_msg),0,(void*)&(L->cin),sizeof(L->cin))<0)
{
ERR_MSG("sendto");
return -1;
}
}
else //删除已下线用户地址信息
{
linklist temp = L->next;
L->next = temp->next;
free(temp);
}
}
}
return 0;
}
//分支线程:从终端获取数据并发送系统消息
void* SndMsg(void* arg) //void* arg = (void*)&info
{
char receive[128]="";
//发送数据
fgets(receive,sizeof(receive),stdin);
receive[strlen(receive)-1]=0;
int sfd = ((struct snd*)arg)->sfd;
struct sockaddr_in sin = ((struct snd*)arg)->sin;
MSG rcv_msg;
sprintf(rcv_msg.text,"system:%s",receive);
if(sendto(sfd,&rcv_msg,sizeof(rcv_msg),0,(void*)&sin,sizeof(sin))<0)
{
ERR_MSG("sendto");
return NULL;
}
printf("系统消息发送成功\n");
}