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;
	
}

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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值