基于UDP实现直播间聊天的功能

news/2024/5/18 12:35:54 标签: udp, 网络协议, 网络, c语言, linux
需求:
	软件划分为用户客户端和主播服务端两个软件client.c和server.c 

用户客户端负责:
    1.接收用户的昵称
    2.接收用户输入的信息,能够将信息发送给服务端
    3.接收服务端回复的数据信息,并完成显示

主播服务端负责:
    1.对所有加入直播间的用户的IP地址和端口实现管理(加入、退出)
    2.当有新的客户端加入时,能够向所有客户端提示:"欢迎 XXX 用户进入直播间"
    3.当有客户端退出时,能够向所有客户端提示:"XXX 离开直播间"
    4.能够实现客户端聊天内容的转发,当某个客户端发送聊天信息时,能够将该信息转给除了该用户之外聊天室内所有其余客户端用户

head.h文件

#ifndef __HEAD_H__
#define __HEAD_H__

#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <dirent.h>
#include <time.h>
#include <pwd.h>
#include <grp.h>
#include <pthread.h>
#include <semaphore.h>
#include <signal.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/ip.h>
#include <netinet/in.h>

struct address
{
	int mark;
	struct sockaddr_in addr;
};

struct msgbuf{
	int type;
	char name[32];
	char text[512];
};

#define MSG_TYPE_START		100
#define MSG_TYPE_END		200
#define MSG_TYPE_CHAT		300
#define RECV_ADDR			"192.168.1.152"
#define RECV_PORT			50000

#endif

client.c

#include "head.h"

char name[32] = {0};
int sockfd = 0;
struct sockaddr_in recvaddr;
pthread_t tid_send;		//发线程
pthread_t tid_recv;		//收线程

void *sendfun(void *arg)
{
	struct msgbuf sendmsg;	
	ssize_t nsize = 0;

	while (1)
	{
		memset(&sendmsg, 0, sizeof(sendmsg));
		sendmsg.type = MSG_TYPE_CHAT;
		sprintf(sendmsg.name, "%s", name);
		gets(sendmsg.text);
		
		if (!strcmp(sendmsg.text, ".quit"))
		{
			sendmsg.type = MSG_TYPE_END;
		}

		nsize = sendto(sockfd, &sendmsg, sizeof(sendmsg), 0, (struct sockaddr *)&recvaddr, sizeof(recvaddr));
		if (-1 == nsize)
		{
			perror("fail to sendto");
			return NULL;
		}

		if (sendmsg.type == MSG_TYPE_END)
		{
			break;
		}
	}

	pthread_cancel(tid_recv);
	
	return NULL;
}

void *recvfun(void *arg)
{
	struct msgbuf recvmsg;
	ssize_t nsize = 0;

	while (1)
	{
		memset(&recvmsg, 0, sizeof(recvmsg));
		nsize = recvfrom(sockfd, &recvmsg, sizeof(recvmsg), 0, NULL, NULL);
		if (-1 == nsize)
		{
			perror("fail to recvfrom");
			return NULL;
		}

		if (recvmsg.type == MSG_TYPE_CHAT)
		{
			printf("%s(%s:%d)>%s\n", recvmsg.name, RECV_ADDR, RECV_PORT, recvmsg.text);
		}
		else if (recvmsg.type == MSG_TYPE_END)
		{
			break;
		}
	}
	
	pthread_cancel(tid_send);

	return NULL;
}

int main(void)
{
	struct msgbuf sendmsg;
	ssize_t nsize = 0;

	recvaddr.sin_family = AF_INET;
	recvaddr.sin_port = htons(RECV_PORT);
	recvaddr.sin_addr.s_addr = inet_addr(RECV_ADDR);

	printf("请输入您的昵称:\n");
	gets(name);

	sockfd = socket(AF_INET, SOCK_DGRAM, 0);
	if (-1 == sockfd)
	{
		perror("fail to socket");
		return -1;
	}
	
	memset(&sendmsg, 0, sizeof(sendmsg));
	sendmsg.type = MSG_TYPE_START;
	sprintf(sendmsg.name, "%s", name);
	nsize = sendto(sockfd, &sendmsg, sizeof(sendmsg), 0, (struct sockaddr *)&recvaddr, sizeof(recvaddr));
	if (-1 == nsize)
	{
		perror("fail to sendto");
		return -1;
	}

	pthread_create(&tid_send, NULL, sendfun, NULL);
	pthread_create(&tid_recv, NULL, recvfun, NULL);

	pthread_join(tid_send, NULL);
	pthread_join(tid_recv, NULL);

	close(sockfd);

	return 0;
}

server.c

#include "head.h"

struct address ClientIpList[100];

int AddClientIp(struct sockaddr_in TmpAddr)
{
	int i = 0;

	for(i = 0;i < 100;++i)
	{
		if(ClientIpList[i].mark == 0)
		{
			ClientIpList[i].addr = TmpAddr;
			ClientIpList[i].mark = 1;
			break;
		}
	}

	return 0;
}
int DelClientIp(struct sockaddr_in TmpAddr)
{
	int i = 0;

	for(i = 0;i < 100;++i)
	{
		if(0 == memcmp(&TmpAddr,&ClientIpList[i].addr,sizeof(TmpAddr)))
		{
			ClientIpList[i].mark = 0;
			break;
		}
	}

	return 0;
}

int BoardcastClientIp(int sockfd,struct sockaddr_in TmpAddr,struct msgbuf TmpMes)
{
	int i = 0;
	ssize_t nsize = 0;
	for(i = 0;i < 100;++i)
	{
		if(ClientIpList[i].mark == 0)
		{
			continue;
		}

		if(0 != memcmp(&TmpAddr,&ClientIpList[i].addr,sizeof(TmpAddr)))
		{
			nsize = sendto(sockfd,&TmpMes,sizeof(TmpMes),0,(struct sockaddr *)&ClientIpList[i].addr,sizeof(ClientIpList[i].addr));
			if(-1 == nsize)
			{
				continue;
			}
		}
	}

	return 0;
}


int main(void)
{
	int sockfd = 0;
	int ret = 0;
	ssize_t nsize = 0;
	struct msgbuf recvmes;
	struct sockaddr_in recvaddr;
	struct sockaddr_in sendaddr;
	socklen_t addrlen = sizeof(sendaddr);

	recvaddr.sin_family = AF_INET;
	recvaddr.sin_port = htons(RECV_PORT);
	recvaddr.sin_addr.s_addr = INADDR_ANY;

	sockfd = socket(AF_INET, SOCK_DGRAM, 0);
	if (-1 == sockfd)
	{
		perror("fail to socket");
		return -1;
	}
	
	ret = bind(sockfd, (struct sockaddr *)&recvaddr, sizeof(recvaddr));
	if (-1 == ret)
	{
		perror("fail to bind");
		return -1;
	}

	memset(&ClientIpList,0,sizeof(ClientIpList));
	while (1)
	{
		memset(&recvmes, 0, sizeof(recvmes));
		nsize = recvfrom(sockfd, &recvmes, sizeof(recvmes), 0,(struct sockaddr *)&sendaddr, &addrlen);
		if (-1 == nsize)
		{
			perror("fail to recvfrom");
			return -1;
		}
		
		
		if(recvmes.type == MSG_TYPE_START)
		{
			AddClientIp(sendaddr);
			recvmes.type = MSG_TYPE_CHAT;
			sprintf(recvmes.text,"%s",recvmes.name);
		}
		else if (recvmes.type == MSG_TYPE_END)
		{
			DelClientIp(sendaddr);
			recvmes.type = MSG_TYPE_CHAT;
			sprintf(recvmes.text,"%s",recvmes.name);
		}
		
		if (recvmes.type == MSG_TYPE_CHAT)
		{
			BoardcastClientIp(sockfd,sendaddr,recvmes);
		}
	}

	close(sockfd);

	return 0;
}

结果:

在这里插入图片描述


http://www.niftyadmin.cn/n/5413621.html

相关文章

990-41产品经理:Essential Skills for Written Communication 书面沟通的基本技能

What is written communication? 什么是书面沟通&#xff1f; In the age of information, there is simply too much to remember. A simple solution is to write it all down. Written communication definition Written communication is making use of the written word…

05.XMLHttpRequest(XHR)的基础使用,查询参数和数据提交

一.XMLHttpRequest - 基础使用 AJAX 是浏览器与服务器通信的技术&#xff0c;采用 XMLHttpRequest 对象相关代码 axios 是对 XHR 相关代码进行了封装&#xff0c;让我们只关心传递的接口参数 学习 XHR 也是了解 axios 内部与服务器交互过程的真正原理 语法如下&#xff1a;…

Unity3D 实现大世界地图的技术原理详解

前言 Unity3D是一款非常强大的游戏引擎&#xff0c;可以用于创建各种类型的游戏&#xff0c;包括大世界地图。在这篇文章中&#xff0c;我们将详细介绍如何使用Unity3D实现大世界地图&#xff0c;并给出相应的技术原理和代码实现。 对惹&#xff0c;这里有一个游戏开发交流小…

蓝桥杯物联网竞赛_STM32L071_11_知识体系的查漏与补缺

太久没学单片机了&#xff0c;再重新过一遍查漏补缺&#xff0c;对其中之前没怎么在意的&#xff0c;而现在又发觉的问题进行再分析与补充 1. debug serial wire是干什么用的 这个东西我勾选不勾选都对我的程序没有什么影响&#xff0c;我很好奇是干什么用的&#xff0c;网上查…

2024年AI辅助研发的技术革新与应用展望

文章目录 每日一句正能量前言AI辅助研发的技术进展全球AI应用呈现出百家争鸣、百花齐放态势&#xff0c;加速向各行各业渗透AI应用显著促进效率提升&#xff0c;“劳动替代低创造性脑力替代”正在加速形成 面临的挑战与机遇未来趋势预测后记 每日一句正能量 要理解这样的自己。…

CAN总线及通讯的工作原理

一、CAN总线 CAN是控制器局域网络(Controller Area Network)的简称&#xff0c; 它是由研发和生产汽车电子产品著称的德国BOSCH公司开发的&#xff0c; 并最终成为国际标准&#xff08;ISO11519&#xff09;&#xff0c;是国际上应用最广泛的现场总线之一。 二、工作原理 …

探索vue框架的世界: 内部、外部样式和内联样式动态绑定的方法

在实际项目中&#xff0c;经常会遇到这样的场景&#xff0c;可以通过逻辑层中设定的变量&#xff0c;在视图层中来呈现不同的样式&#xff0c;那么这种动态绑定样式的方式如何实现呢&#xff1f; 本篇文章&#xff0c;博主将和大家分享动态绑定内联样式style 和 动态绑定内部和…

Bytebase 签约合思,覆盖多云数据库变更发布,数据访问控制,安全治理的全生命周期,确保符合合规审计要求

在数字化快速发展时代&#xff0c;有效的规范数据库管理对企业安全运营至关重要。近日&#xff0c;数据库 DevOps 团队协同管理工具 Bytebase 签约费控领域领军企业合思&#xff0c;旨在全面优化数据库操作管理&#xff0c;收口全体员工的变更和查询操作&#xff0c;以提高整体…