udp协议实现组播功能
/**************************************************************************************************************************************
*
* file name:multicast.c
* author :momolyl@126.com
* date :2024/06/04
* brief :小组实现,小组中的每位成员都需要设计程序,要求程序可以加入到一个多播组中并等待服务器发送数据包,
* 并且程序还需要具有发送功能,如果收到数据包则把消息内容输出到终端,
* 消息内容格式 [消息来源IP 消息时间 ] : 消息内容
* note :运行多播可执行文件 ./multicast 服务器端口 多播组地址
*
* CopyRight (c) 2024 momolyl@126.com All Right Reseverd
* ***********************************************************************************************************************************/
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/udp.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>
#include <sys/types.h>
#include <time.h>
#include <stdlib.h>
#include <stdio.h>
#include <strings.h>
#include <unistd.h>
#include <time.h>
// 接受消息线程
void *pthreadrecv_task(void *argv)
{
// 1.创建UDP套接字
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 2.需要先绑定服务器的端口和地址
struct sockaddr_in host_addr;
host_addr.sin_family = AF_INET; // 协议族,是固定的
host_addr.sin_port = htons(atoi("9999")); // 目标端口,必须转换为网络字节序
host_addr.sin_addr.s_addr = htonl(INADDR_ANY); // 目标地址 "192.168.64.xxx" 已经转换为网络字节序 INADDR_ANY
bind(udp_socket, (struct sockaddr *)&host_addr, sizeof(host_addr));
// 3.调用recvfrom等待接收数据,并且接收客户端的网络信息
char buf[128] = {0};
struct sockaddr_in client;
socklen_t client_len = sizeof(client);
// 2.获取当前系统时间
const char *weekday[] = {"日", " 一", " 二", "三", "四", "五", "六"};
char timebuf[128] = {0};
time_t CurrentTime;
while (1)
{
time(&CurrentTime);
struct tm *time = localtime(&CurrentTime);
sprintf(timebuf, "%d年 %02d月 %02d日 星期%s %02d:%02d:%02d", time->tm_year + 1900, time->tm_mon + 1, time->tm_mday,
weekday[time->tm_wday], time->tm_hour, time->tm_min, time->tm_sec);
recvfrom(udp_socket, buf, sizeof(buf), 0, (struct sockaddr *)&client, &client_len); // 默认会阻塞
printf("[消息来源:%s 当前时间:%s]:%s\n", inet_ntoa(client.sin_addr), timebuf, buf);
bzero(buf, sizeof(buf));
bzero(timebuf, sizeof(timebuf));
}
}
int main(int argc, char *argv[])
{
char timebuf[128] = {0};
pthread_t pthreadrecv;
pthread_create(&pthreadrecv, NULL, pthreadrecv_task, NULL);
// 检查参数有效性(多播组地址:224.6.6.6 端口:9999)
if (argc != 3)
{
fprintf(stderr, "argument is invaild ,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 1.创建UDP套接字
int udp_socket = socket(AF_INET, SOCK_DGRAM, 0);
if (udp_socket == -1)
{
fprintf(stderr, "udp socket error,errno:%d,%s\n", errno, strerror(errno));
exit(1);
}
// 加入多播组
struct ip_mreqn mutipcast;
struct in_addr imr_multiaddr;
mutipcast.imr_multiaddr.s_addr = inet_addr("224.6.6.6");
mutipcast.imr_address.s_addr = inet_addr("192.168.64.230");
mutipcast.imr_ifindex = 0;
setsockopt(udp_socket, IPPROTO_IP, IP_ADD_MEMBERSHIP, (void *)&mutipcast, sizeof(mutipcast));
// 设置广播属性
int optval = 1;
setsockopt(udp_socket, SOL_SOCKET, SO_BROADCAST, (void *)&optval, 4);
// 2.向目标主机发送消息,需要设置目标端口和目标地址
char buf[128] = "HELLO WORLD";
struct sockaddr_in dest_addr;
dest_addr.sin_family = AF_INET; // 协议族,是固定的
dest_addr.sin_port = htons(atoi(argv[1])); // 服务器端口,必须转换为网络字节序
dest_addr.sin_addr.s_addr = inet_addr(argv[2]); // 服务器地址 "192.168.64.xxx"
while (1)
{
// 3.向多播组发送内容
sendto(udp_socket, buf, strlen(buf), 0, (struct sockaddr *)&dest_addr, sizeof(dest_addr));
sleep(5);
}
return 0;
}
运行结果:
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!