0812 练习

在这里插入图片描述
头文件:

#ifndef __CHAT_H__
#define __CHAT_H__
 
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <signal.h>
#define IP "192.168.31.117"

//打印错误新的宏函数
#define ERR_MSG(msg1)  do{\
	fprintf(stderr, " __%d__ ", __LINE__);\
	perror(msg1);\
}while(0)
 
 
//存储客户端地址信息的结构体说明
typedef struct Node
{
	struct sockaddr_in cin; 
    struct Node *next;         //指针域,记录下个结点的地址
}linkList;

//定义数据包的结构体
struct USER_MSG
{
	char type;    // 1.登录 2.聊天 3.下线
	char username[20];
	char text[516];
};

//遍历链表并且发送登录信息
void list_login(linkList*user,struct USER_MSG msg,int sfd);


//将客户端信息头插进链表
int list_insert_head(linkList *user,struct sockaddr_in cin);

//遍历并发送聊天信息
void list_msg(linkList*user,struct USER_MSG msg,int sfd,struct sockaddr_in cin);


//遍历并发送下线信息
void list_exit(linkList*user,struct USER_MSG msg,int sfd,struct sockaddr_in cin);


#endif

调用函数:

#include"chat.h"


//将客户端信息头插进链表
int list_insert_head(linkList *user,struct sockaddr_in cin)
{
	linkList *p=(linkList*)malloc(sizeof(linkList));
	if(NULL==p)
	{
		printf("节点申请失败\n");
		return -1;
	}
	p->cin=cin;
	p->next=NULL;
	//头插
	p->next=user->next;
	user->next=p;
	return 0;
}



//遍历链表并且发送登录信息
void list_login(linkList*user,struct USER_MSG msg,int sfd)
{
	linkList *q=user;
	sprintf(msg.text,"%s","登录成功");
	while(q->next!=NULL)	
	{
		q=q->next;
		if(sendto(sfd,&msg,sizeof(msg),0,(struct sockaddr*)&(q->cin),sizeof(q->cin))<0)
		{
			ERR_MSG("sendto");
			return;
		}
	}
}


//遍历并发送聊天信息
void list_msg(linkList*user,struct USER_MSG msg,int sfd,struct sockaddr_in cin)
{
	linkList*q=user;
	while(q->next!=NULL)
	{
		q=q->next;
		if((q->cin).sin_port==cin.sin_port)
		{
			if(sendto(sfd,&msg,sizeof(msg),0,(struct sockaddr*)&(q->cin),sizeof(q->cin))<0)
			{
				ERR_MSG("sendto");
				return;
			}
		}
	
	}
}

//遍历并发送下线信息
void list_exit(linkList*user,struct USER_MSG msg,int sfd,struct sockaddr_in cin)
{
	linkList*q=user;
	sprintf(msg.text,"%s","退出聊天");
	while(q->next!=NULL)
	{
		q=q->next;
		if((q->cin).sin_port==cin.sin_port)
		{
			if(sendto(sfd,&msg,sizeof(msg),0,(struct sockaddr*)&(q->cin),sizeof(q->cin))<0)
			{
				ERR_MSG("sendto");
				return;
			}		
		}
		else
		{
			
			if(q==user)
			{
				user=user->next;
				free(q);	
				q=NULL;
			}
			else 
			{
				linkList*p=q->next;
				q->cin=p->cin;
				q->next=p->next;
				free(p);
				p=NULL;
			}

		}

	}
}

客户端:

#include"chat.h"


int main(int argc, const char *argv[])
{
	
	//创建报式套接字
	int sfd = socket(AF_INET, SOCK_DGRAM, 0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}
	
	//填充服务器的IP地址以及端口号
	struct sockaddr_in sin;
	sin.sin_family      = AF_INET;
	sin.sin_port        = htons(8888);
	sin.sin_addr.s_addr = inet_addr(IP);


	
	
	

	//用户登录
	struct USER_MSG msg;
	msg.type=1;
	printf("请输入用户名:");
	fgets(msg.username,20,stdin);
	msg.username[strlen(msg.username)-1]=0;
	
	//发送登录请求
	if(sendto(sfd,&msg,sizeof(msg),0,NULL,0)<0)
	{
		ERR_MSG("sendto");
		return -1;
	}
	
	//创建父子进程
	pid_t pid=fork();
	if(0==pid)
	{
		//子进程负责发送请求
		while(1)
		{
			fgets(msg.text,516,stdin);
			msg.text[strlen(msg.text)-1]=0; //向数据包写入信息
			
			//判断是否下线
			if(strcasecmp(msg.text,"quit"))
			{
				msg.type=2;
				if(sendto(sfd,&msg,sizeof(msg),0,NULL,0)<0)
				{
					ERR_MSG("sendto");     //不下线执行聊天
					return -1;
				}
			}

			else
			{
				msg.type=3;
				exit(0);        //下线直接退出
			}
		}
	
	}
	else if(pid>0)
	{
		//父进程接收
		while(1)
		{
			bzero(&msg,sizeof(msg));
			if(recvfrom(sfd,&msg,sizeof(msg),0,NULL,NULL)<0)
			{
				ERR_MSG("recvfrom");
				return -1;
			}
			printf("%s:%s\n",msg.username,msg.text);
		}
	}
	else
	{
		ERR_MSG("fork");
		return -1;
	}
	close(sfd);
	return 0;
}

服务端:

#include"chat.h"


int main(int argc, const char *argv[])
{
	

	//创建报式套接字
	int sfd = socket(AF_INET, SOCK_DGRAM, 0);
	if(sfd < 0)
	{
		ERR_MSG("socket");
		return -1;
	}
 
 
	//填充服务器的IP地址以及端口号
	struct sockaddr_in sin;
	sin.sin_family      = AF_INET;
	sin.sin_port        = htons(8878);
	sin.sin_addr.s_addr = inet_addr(IP);
 
 
	//绑定服务器的地址信息结构体
	if(bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)        
	{
		ERR_MSG("bind");
		return -1;
	}
	printf("bind success\n");

	//定义客户端的网络信息结构体
	struct sockaddr_in cin;
	socklen_t addrlen=sizeof(cin);
	//创建父子进程
	pid_t pid=fork();
	if(pid>0)   //父进程负责接收客户端的请求
	{  
		//创建存储客户端信息的链表						
		linkList* user=(linkList*)malloc(sizeof(linkList));
		if(NULL==user)
		{
			printf("创建失败\n");
			return -1;
		}

		user->next=NULL;
		printf("create success\n");
		struct USER_MSG msg;
		while(1)
		{
			bzero(&msg,sizeof(msg));
			if(recvfrom(sfd,&msg,sizeof(msg),0,(struct sockaddr*)&cin,&addrlen)<0)
			{
				ERR_MSG("recvfrom");
				return -1;
			}
			if(msg.type==1)
			{
				list_login(user,msg,sfd);
				list_insert_head(user,cin);
			}
			else if(msg.type==2)
			{
				list_msg(user,msg,sfd,cin);
			}
			else if(msg.type==3)
			{
				list_exit(user,msg,sfd,cin);
			}
		}
	
	}	
	
	return 0;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值