UDP组播简单的代码实现

    通过广播可以很方便地实现发送数据包给局域网中的所有主机。但广播同样存在一些问题,例如,频繁地发送广播包造成所有主机数据链路层都会接收并交给上层协议处理,也容易引起局域网的网络风暴。

    当发送组播数据包时,只有加入指定多播组的主机数据链路层才会处理,其他主机在数据链路层会直接丢掉收到的数据包。换句话说,我们可以通过组播的方式和指定的若干主机通信。

    D类地址又被称为组播地址。每一个组播地址代表一个多播地址。

    下面是简单的UDP组播代码。

     服务端:

/*multicast_recv.c*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include<sys/types.h>
#include <sys/socket.h>
#include<netinet/in.h>
#include <arpa/inet.h>
 
#define N 64
 
int main(int argc, const char *argv[])
{
    int sockfd;
    char buf[N] = "0";
    struct sockaddr_in myaddr, peeraddr;
	struct ip_mreq mreq;
	socklen_t peerlen = sizeof(peeraddr);
	
    if((sockfd = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
    {
        perror("socket");
        exit(-1);
    }
	
    bzero(&mreq, sizeof(mreq));
    mreq.imr_multiaddr.s_addr = inet_addr(argv[1]);
	mreq.imr_interface.s_addr = htonl(INADDR_ANY);
	
	if(setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0)
	{
		perror("setsockopt");
		exit(-1);
	}
	
 
    if((sockfd = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
    {
        perror("socket");
        exit(-1);
    }
    bzero(&myaddr, sizeof(myaddr));
    myaddr.sin_family = PF_INET;
    myaddr.sin_port = htons(atoi(argv[2]));
    myaddr.sin_addr.s_addr = inet_addr(argv[1]);
	
	if(bind(sockfd, (struct sockaddr *)&myaddr, sizeof(myaddr)) < 0)
	{
		perror("fail to bind\n");
		exit(-1);
	}
	
	while(1)
	{
		recvfrom(sockfd, buf, N, 0, (struct sockaddr *)&peeraddr, &peerlen);
		printf("%s : %d  %s\n", inet_ntoa(peeraddr.sin_addr), ntohs(peeraddr.sin_port), buf);
	}
	return 0;
}

 客户端:

   

/*  multicast_send.c
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define N 64
typedef struct sockaddr SA;

int main(int argc,char* argv[])
{
	int sockfd;
	char buf[N]="This is a multicast package\n";
	struct sockaddr_in dstaddr;
	
	if(argc < 3)
	{
		printf("Usage:<%s> <ip> <port>\n",argv[0]);
		return -1;
	}	

	if((sockfd = socket(PF_INET,SOCK_DGRAM,0)) == -1)
	{
		perror("failed to socket");
		exit(-1);
	}

	bzero(&dstaddr,sizeof(dstaddr));
	dstaddr.sin_family = PF_INET;
	dstaddr.sin_port = htons(atoi(argv[2]));
	dstaddr.sin_addr.s_addr = inet_addr(argv[1]);
	
	while(1)
	{
		sendto(sockfd,buf,N,0,(SA *)&dstaddr,sizeof(dstaddr));
		sleep(1);
	}	
	return 0;
	
}

如上图所示,是一个服务端加入组播地址,两个客户端给该组播地址发送数据。

广播应该是两个服务端加入该组播地址,一个客户端发送的数据,这两个服务端都可以接收到数据。这个可以自己动手调试下。

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
UDP是通过在IP层使用组地址来实现的,组地址是D类IP地址(224.0.0.0至239.255.255.255)。发送方将数据发送到组地址,而所有已加入该组地址的接收方都会收到该数据。 UDP实现代码需要借助多套接字(multicast socket)。发送方需要使用setsockopt()函数将套接字加入到组地址,使用sendto()函数发送数据;接收方也需要使用setsockopt()函数将套接字加入到组地址,并使用recvfrom()函数接收数据。 以下是一个简单UDP实现代码示例: 发送方: ``` #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <string.h> #include <stdio.h> int main() { int sockfd; struct sockaddr_in addr; char buf[1024] = "Hello, multicast!"; sockfd = socket(AF_INET, SOCK_DGRAM, 0); memset(&addr, 0, sizeof(addr)); addr.sin_family = AF_INET; addr.sin_port = htons(8000); inet_pton(AF_INET, "239.0.0.1", &addr.sin_addr); int flag = 1; setsockopt(sockfd, IPPROTO_IP, IP_MULTICAST_LOOP, &flag, sizeof(flag)); sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr*)&addr, sizeof(addr)); close(sockfd); return 0; } ``` 接收方: ``` #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <string.h> #include <stdio.h> int main() { int sockfd; struct sockaddr_in addr; char buf[1024]; sockfd = socket(AF_INET, SOCK_DGRAM, 0); memset(&addr, 0, sizeof(addr)); addr.sin_family = AF_INET; addr.sin_port = htons(8000); addr.sin_addr.s_addr = htonl(INADDR_ANY); bind(sockfd, (struct sockaddr*)&addr, sizeof(addr)); struct ip_mreq mreq; inet_pton(AF_INET, "239.0.0.1", &mreq.imr_multiaddr); mreq.imr_interface.s_addr = htonl(INADDR_ANY); setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)); int len = recvfrom(sockfd, buf, sizeof(buf), 0, NULL, 0); buf[len] = '\0'; printf("Received message: %s\n", buf); close(sockfd); return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值