组播

  1. 广播方式是发送给同一网段下的所有主机,过多的广播会占用大量网络带宽,造成广播风暴,影响正常通讯。
  2. 组播(多播),加入到多播组中的主机才可以收发消息。
  3. 多播方式既可以给多个主机发送,又能避免造成过多负载。
  4. 每台主机要到传输层才能判断组播是否要处理。
  5. 组播地址:D类 IP地址:224.0.0.0~239.255.255.255

组播接收端(服务器)

  1. 创建报式套接字(socket)
  2. 加入多播组(setsockopt)
  3. 绑定组播IP(例如:224.1.2.3或者0.0.0.0)和端口号(bind)
  4. 等待接收数据(recvfrom)
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#define PORT 2222
#define IP   "224.1.2.3" 
int main(int argc, const char *argv[])
{
    //1.创建报式套接字
    int sfd = socket(AF_INET, SOCK_DGRAM, 0);
    if(sfd < 0)
    {
        perror("socket");
        return -1;
    }
    //2.加入多播组
    struct ip_mreq mq;
    mq.imr_multiaddr.s_addr = inet_addr(IP); //组播ip
    mq.imr_interface.s_addr = inet_addr("0.0.0.0"); //本机ip
    if(setsockopt(sfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mq, sizeof(mq))<0)
    {
        perror("setsockopt");
        return -1;
    }
    printf("----加入多播组成功----\n");
    //2.绑定服务器ip和端口号
    struct sockaddr_in sin;
    sin.sin_family = AF_INET;
    sin.sin_port = htons(PORT);
    sin.sin_addr.s_addr = inet_addr(IP);
    if(bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) < 0)
    {
        perror("bind");
        return -1;
    }
    char buf[128] = "";
    struct sockaddr_in cin;
    socklen_t addrlen = sizeof(cin);
    //3.接收
    while(1)
    {
        bzero(buf, sizeof(buf));
        if(recvfrom(sfd, buf, 128, 0, (struct sockaddr*)&cin, &addrlen) < 0)
        {
            perror("recvfrom");
            return -1;
        }
        printf("[%s:%d]:%s\n", (char*)inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);
    }
    //4.关闭套接字
    close(sfd);
    return 0;
}

组播发送端(客户端)

  1. 创建报式套接字(socket)
  2. 绑定客户端信息,非必须(bind)
  3. 填充接收方信息,地址为组播地址(224.0.0.0~239.255.255.255)和端口号
  4. 发送数据(sendto)
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#define PORT 2222
#define IP   "224.1.2.3" 
int main(int argc, const char *argv[])
{
    //1.创建报式套接字
    int sfd = socket(AF_INET, SOCK_DGRAM, 0);
    if(sfd < 0)
    {
        perror("socket");
        return -1;
    }
    //2.填充服务器ip和端口号
    struct sockaddr_in sin;
    sin.sin_family = AF_INET;
    sin.sin_port = htons(PORT);
    sin.sin_addr.s_addr = inet_addr(IP);
    char buf[128] = "hello---";
    socklen_t addrlen = sizeof(sin);
    //3.发送
    while(1)
    {
        bzero(buf, sizeof(buf));
        printf("请输入>>>");
        fgets(buf, 128, stdin);
        buf[strlen(buf)-1] = 0;
        if(sendto(sfd, buf, 128, 0, (struct sockaddr*)&sin, addrlen)<0)
        {
            perror("sendto");
            return -1;
        }
    }
    //4.关闭套接字
    close(sfd);
    return 0;
}
Last modification:2021 年 04 月 19 日 14 : 53 : 02