网络编程(19)—— 多播的编程实现

本文详细介绍了如何使用UDP协议实现多播技术,包括发送端和接收端的具体实现过程,并提供了完整的示例代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、引言       

        多播,也叫组播,是使用UDP协议传递数据的一种方式。发送数据的主机向一个计算机组发送数据,所有注册在该计算机组中的计算机都能接受到该数据。我们经常进行的视频会议,就是用多播实现的。在网络中,一般通过路由器可实现该功能。首先,发送数据的主机发送一组数据,然后到达支持多播功能的路由器后,路由器会进行复制,将数据包复制后发给本组的其他计算机。

        在软件中实现多播,需要将目标地址设置成任意D类ip地址(224.0.0.0~239.255.255.255),然后在发送主机中设置发送的socket选项,设置TTL。TTL是time_live的简称,意为生存时间,它的含义可以理解为数据包可以经过路由器的最大数。它的数值,每经过一个路由器就是减1,当减到0时,该数据包就不会再传递,因此它决定了数据包传送的距离。

TTL设置方法如下:

int send_sock; 
int time_live=64; 
... 
send_sock=socket(AF_INET,SOCK_DGRAM,0); 
setsockopt(send_sock,IPPROTO_IP,IP_MULTICAST_TTL,(void*)&time_live,sizeof(time_live));
send_sock就是我们要设置的socke对象。我们利用setsockopt函数设置支持多播的socket选项IPPROTO_TP
 
         然后还需要在接收主机中,将主机socket加入对应的接收计算机组。 接收端利用ip_mreq类型的结构体设置多播的地址和主机地址,ip_mreq的imr_multiaddr.s_addr成员接收的是多播的地址,imr_interface.s_addr成员接收的是本地的地址。然后再利用setsockopt()函数设置接收端主机的socket即可。
int recv_sock; 
struct ip_mreq join_adr; 
... 
recv_sock=socket(AF_INET,SOCK_DGRAM,0); 
... 
join_adr.imr_multiaddr.s_addr="多播组地址信息"; 
join_adr.imr_interface.s_addr="加入多播组的主机地址信息"; 
setsockopt(recv_sock,IPPROTO_IP,IP_ADD_MEMBERSHIP,(void*)&join_adr,sizeof(join_adr));

  以下示例中,我们将利用多播实现:发送端读取本地文件news.txt的内容,然后通过多播的方式进行发送,这样凡是加入该多播组的所有主机都能接收到发送端发送的数据。

二、发送端

        发送端的示例代码如下:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/socket.h>
#include<arpa/inet.h>

#define BUF_SIZE 30
void error_handling(char* message);
int main(int argc,char* argv[])
{
    int target_sock;
    struct sockaddr_in target_addr;
    FILE* fp;
    char buf[BUF_SIZE];
    int time_live=64;
    if(argc!=3)
    {
        printf("Usage %s <address> <port> \n",argv[0]);
        exit(1);
    }
    target_sock=socket(AF_INET,SOCK_DGRAM,0);
    
    target_addr.sin_family=AF_INET;
    target_addr.sin_addr.s_addr=inet_addr(argv[1]);
    target_addr.sin_port=htons(atoi(argv[2]));
    
   fp=fopen("news.txt","r");
   if(fp==NULL)
   {
        error_handling("fopen error!");
   }

    setsockopt(target_sock,IPPROTO_IP,
                    IP_MULTICAST_TTL,(void*)&time_live,sizeof(time_live));

    while(!feof(fp))
    {
        fgets(buf,BUF_SIZE,fp);
        sendto(target_sock,buf,strlen(buf),
                        0,(struct sockaddr*)&target_addr,sizeof(target_addr));
        sleep(2);
    }

    fclose(fp);
    close(target_sock);
    return 0;
}
void error_handling(char* message)
{
    fputs(message,stderr);
    fputc('\n',stderr);
    exit(1);
}
第28行,打开news.txt文件,获得文件指针。

第34行,利用setsockopt设置socket的TTL,而多播组的地址是我们通过main函数的参数传进去的,无需特别的设置,只需指明D类的多播地址即可。

我们可以通过下面的方法调用服务端,233.1.1.1就是多播的地址。

./sender 223.1.1.1 9190 

二 接收端

         接收端的示例代码如下:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<arpa/inet.h>
#include<sys/socket.h>

#define BUF_SIZE 30

void error_handling(char* message);

int main(int argc,char* argv[])
{
    int recv_sock;
    struct sockaddr_in recv_addr;
    char buf[BUF_SIZE];
    struct ip_mreq join_adr;
    int str_len;
    if(argc!=3)
    {
        printf("Usage %s <adress> <port>\n",argv[0]);
        exit(1);
    }
    recv_sock=socket(AF_INET,SOCK_DGRAM,0);

    recv_addr.sin_family=AF_INET;
    recv_addr.sin_addr.s_addr=htonl(INADDR_ANY);
    recv_addr.sin_port=htons(atoi(argv[2]));

    if(bind(recv_sock,(struct sockaddr*)&recv_addr,sizeof(recv_addr))==-1)
            error_handling("bind error");
    
    join_adr.imr_multiaddr.s_addr=inet_addr(argv[1]);
    join_adr.imr_interface.s_addr=htonl(INADDR_ANY);

    setsockopt(recv_sock,IPPROTO_IP,
                    IP_ADD_MEMBERSHIP,(void*)&join_adr,sizeof(join_adr));
    
    while(1)
    {
       str_len=recvfrom(recv_sock,buf,BUF_SIZE-1,MSG_DONTWAIT,NULL,0);
       printf("接收到 %d 字节。\n",str_len); 
       if(str_len<=0)
                break;

        buf[str_len]=0;
        fputs(buf,stdout);
    }
    
    close(recv_sock);

    return 0;
}

void error_handling(char* message)
{
    fputs(message,stderr);
    fputc('\n',stderr);
    exit(1);
}

第17行,我们声明了一个ip_mreq类型的对象,用来存放多播的组地址和本机的IP地址。

第33、34行分别将多播地址和本机地址传给ip_mreq对象的对应成员。需要注意的是我们这里的多播地址是通过main函数的第二个参数传过来的。

第36行,通过setsockopt()函数使本机加入多播组。


        客户端的用法如下:

./recver 223.1.1.1 9190 
        223.1.1.1就是我们要加入的多播组。


Note:

1、因为TCP协议是有连接的,属于点对点通信,多播(组播)只支持UDP协议。

2、多播接收端和发送端的端口号要一致。


Github位置:

https://github.com/HymanLiuTS/NetDevelopment

克隆本项目:

git clone git@github.com:HymanLiuTS/NetDevelopment.git

获取本文源代码:

git checkout NL19


评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值