「多播」WinSock下基于IP协议的多播实现示例

2024-05-01 03:18

本文主要是介绍「多播」WinSock下基于IP协议的多播实现示例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

多播的含义

 

“多播”亦称“多点传送”(Multicasting),是一种让数据从一个成员送出,然后复制给其他多个成员的技术。

 

        多播通信具有两个层面的重要特征:控制层面和数据层面。其中,“控制层面”(Control Plane)定义了组成员的组织方式;而“数据层面”(Data Plane)决定了在不同的成员之间,数据如何传送。这两方面的特征既可以是“有根的”(Rooted),也可以是“无根的”(Nonrooted)。

        在一个“有根的”控制层面内,存在着一个特殊的多播组成员,叫作 c_root(控制根,或根节点)。而剩下的每个组成员都叫作c_leaf(控制叶,或叶节点)。大多数情况下,c_root 需负责多播组的建立,其间涉及到建立同任意数量的 c_leaf 的连接。而在某些特殊情况下,c_leaf 则可在以后的某个时间申请加入一个特定的多播组(或者说,取得那个组的成员资格)。要注意的是,对任何一个具体的组来说,都只能存在一个根节点。ATM协议便是“有根控制层面”的典型例子。
        而对一个“无根的”控制层面来说,它则允许任何人加入一个组,其间不存在任何例外。在这种情况下,所有组成员均为 c_leaf 节点(叶节点)。每个成员都有权加入一个多播组。IP多播便是无根控制层面的一个典型例子。

 

        数据层面也存在着“有根的”和“无根的”两种形式。对一个有根数据层面而言,它有一个参与者叫作 d_root(数据根,或根节点)。数据传输只能在 d_root 和多播会话的其他所有成员之间进行。显然,那些成员是 d_leaf(数据叶,或叶节点)。这种传输既可单向进行,亦可双向进行。但既然是一个有根数据层面,便暗示着出自一个 d_leaf 叶节点的数据只会被 d_root 根节点接收到;而自 d_root发出的数据却可由每个 d_leaf 收到。ATM也是“有根数据层面”的一个典型例子。

        在一个无根数据层面上,所有组成员都能将数据发给组内的其他所有成员。从一个组成员发出的数据块会投递给其他所有成员,同时所有接收者都能回送数据。至于谁能接收或发送数据,则不存在任何限制。同样地, IP多播采用的是数据层面上的“无根”通信方式。

 

IP多播

 

        IP多播通信需要依赖一个特殊的地址组,名为“多播地址”。我们正是用这个组地址对一个指定的组进行命名。举个例子来说,假定五个节点都想通过IP多播,实现彼此间的通信,它们便可加入同一个组地址。全部加入之后,由一个节点发出的任何数据均会一模一样地复
制一份,发给组内的每个成员,甚至包括始发数据的那个节点。

        多播IP地址是一个D类IP地址,范围在224.0.0.0239.255.255.255之间。但是,其中还有许多地址是为特殊用途而保留的。
比如,224.0.0.0根本没有使用(也不能使用),224.0.0.1代表子网内的所有系统(主机),而224.0.0.2代表子网内的所有路由器。上述最后两个特殊地址只能由IGMP协议使用。

 

        对于多播的简介就到此结束,如果还想了解得更详细,那就只有找本书来慢慢啃了。

 

程序示例

 

        下面的程序演示了如何建立IP多播,并可以接受和发送多播消息。

 

 

#pragma once

#pragma comment(lib, "ws2_32.lib")

 

#include "targetver.h"

 

#include <winsock2.h>

#include <ws2tcpip.h>

#include <process.h>

#include <stdio.h>

 

#define MCASTADDR     "234.5.6.7"   // 加入的多播组IP

#define MCASTPORT      34567        // 加入的多播组所使用的端口

 

void do_send(void* arg);

void do_read(void* arg);

 

int main(int argc, char **argv)

{

    const int   on = 1;  // 用于指定本地同一端口是否允许被多个套接口绑定,非零表示允许

    const int   routenum = 10;  // 用于指定多播数据包的TTL

    const int   loopback = 1;   // 用于指定多播数据包是否回馈,非零表示允许

    WSAData     wsaData;

    SOCKET      server,

                sockM;

    sockaddr_in local,

                remote;

    int         ret ;

 

    if( WSAStartup(MAKEWORD(2,0), &wsaData) != 0 )

    {

        printf("WSAStartup 调用出错./n");

        return 0;

    }

 

    // SOCK_DGRAM 创建一个UDP套接口

    // WSA_FLAG_OVERLAPPED 套接字支持重叠 I/O 操作

    // WSA_FLAG_MULTIPOINT_C_LEAF 套接字支持 c_leaf(控制叶) 的多点会议

    // WSA_FLAG_MULTIPOINT_D_LEAF 套接字支持 d_leaf(数据叶) 的多点会议

    if ((server = WSASocket(AF_INET, SOCK_DGRAM, 0, NULL, 0,

                            WSA_FLAG_MULTIPOINT_C_LEAF

                            | WSA_FLAG_MULTIPOINT_D_LEAF

                            | WSA_FLAG_OVERLAPPED)) == INVALID_SOCKET)

    {

        printf("创建套接字失败,错误代码: %d/n",WSAGetLastError());

        return -1;

    }

 

    // SO_REUSEADDR 设置是否允许本地同一端口被多个套接口绑定

    ret = setsockopt(server, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on));

    if( ret == SOCKET_ERROR )

    {

        closesocket(server);

        WSACleanup();

        printf("套接口选项设置(SO_REUSEADDR)失败,错误代码: %d/n",WSAGetLastError());

        return 0;

    }

    // IP_MULTICAST_TTL 设置多播数据包的转发范围(TTL,Time To Live)

    ret = setsockopt(server,IPPROTO_IP,IP_MULTICAST_TTL,(char*)&routenum,sizeof(routenum));

    if( ret == SOCKET_ERROR )

    {

        closesocket(server);

        WSACleanup();

        printf("套接口选项设置(IP_MULTICAST_TTL)失败,错误代码: %d/n",WSAGetLastError());

        return 0;

    }

    // IP_MULTICAST_LOOP 禁止或允许回馈多播数据包

    ret = setsockopt(server,IPPROTO_IP,IP_MULTICAST_LOOP,(char*)&loopback,sizeof(loopback));

    if( ret == SOCKET_ERROR )

    {

        closesocket(server);

        WSACleanup();

        printf("套接口选项设置(IP_MULTICAST_LOOP)失败,错误代码: %d/n",WSAGetLastError());

        return 0;

    }

    // 初始化

    memset(&local, 0, sizeof(local));

    local.sin_family = AF_INET;

    local.sin_port = htons(MCASTPORT);  // 本地端口

    local.sin_addr.S_un.S_addr = INADDR_ANY;  // 本地全部地址

    // 绑定套接口

    ret = bind(server, (sockaddr*)(&local), sizeof(local));

    if( ret == SOCKET_ERROR )

    {

        closesocket(server);

        WSACleanup();

        printf("套接口绑定失败,错误代码: %d/n",WSAGetLastError());

        return 0;

    }

    // 初始化

    memset(&remote, 0, sizeof(remote));

    remote.sin_family      = AF_INET;

    // 对有根控制方案,该地址指定了欲邀请加入的客户机

    // 对无根控制方案,该地址指定了要加入的多播组

    remote.sin_port        = htons(MCASTPORT);

    remote.sin_addr.S_un.S_addr = inet_addr(MCASTADDR);

    // 加入多播组

    // JL_BOTH 收发兼并

    // JL_SENDER_ONLY 只发

    // JL_RECEIVER_ONLY 只读

    // sockM 返回的套接字描述符取决于输入套接字(server)

    // 对于异步操作,除非加入操作完成,否则返回的套接字描述符是无用的

    // 对于异步操作下的有根方案,若非原来的套接字(server)接收到一个对应的FD_CONNECT通知,

       // 否则返回的描述符便是无效的

    // sockM 主要用于根节点对叶节点的维护管理

    sockM = WSAJoinLeaf(server,(SOCKADDR *)&remote,sizeof(remote),NULL,NULL,NULL,NULL,JL_BOTH);

    if(sockM == INVALID_SOCKET)

    {

        closesocket(server);

        WSACleanup();

        printf("加入多播组失败,错误代码: %d/n",WSAGetLastError());

        return 0;

    }

 

    //创建了两个线程,一个读用户输入并发送,一个读多播组数据

    HANDLE hHandle[2];

    hHandle[0] = (HANDLE)_beginthread(do_send,0,(void*)server);

    hHandle[1] = (HANDLE)_beginthread(do_read,0,(void*)server);

    // 主线程挂起,等待hHandle[0]线程的信号状态发生变化

    // INFINITE 等待时间无限

    // 如果用户输入结束,hHandle[0]线程被释放,该函数检测到信号变化,便返回结果

    WaitForSingleObject(hHandle[0], INFINITE);

    closesocket(server);

    WSACleanup();

    printf("退出多播组,程序结束. ");

    return 0;

}

 

void do_send(void* arg)

{

    const char  end[] = "end",

                END[] = "END";

    SOCKET    server = (SOCKET)arg;

    char      SsendData[1024] = "";

    unsigned int i;

    sockaddr_in remote;

 

    memset(&remote, 0, sizeof(sockaddr_in));

    remote.sin_family = AF_INET ;

    remote.sin_port = htons(MCASTPORT);

    remote.sin_addr.s_addr = inet_addr(MCASTADDR);

    while(TRUE) //读取用户输入直到用户输入"end"

    {

        printf("请输入信息,按回车发送,输入/"end/"退出:/n");

        for(i = 0; i < 1024;i++) // 读取用户输入

        {

            SsendData[i] = getchar();

            // 读到回车(CR)或换行符(LF),便停止读取

            if ((SsendData[i] == (char)0x0A) || (SsendData[i] == (char)0x0D))

                break;

        }

        // 不发送空消息

        if ((SsendData[0] == (char)0x0A) || (SsendData[0] == (char)0x0D))

            continue;

        // 判断退出信息

        if ((SsendData[0] == end[0] || SsendData[0] == END[0]) &&

            (SsendData[1] == end[1] || SsendData[1] == END[1]) &&

            (SsendData[2] == end[2] || SsendData[2] == END[2]))

            break;

        // 发送用户输入的数据到多播组

        // strlen(SsendData) 不包括结尾字符

        sendto(server, SsendData, strlen(SsendData), 0, (sockaddr*)(&remote), sizeof(sockaddr_in));

        // 清除数据,为了下一次发送

        memset(&SsendData, 0, strlen(SsendData));

    }

}

 

void do_read(void* arg)

{

    SOCKET    server = (SOCKET)arg;

    char      ReadData[1024];

    int       ret;

    sockaddr_in client;

    int         clientLen;

    while(TRUE) //一直读取直到主线程终止

    {

        // 清除IP信息缓存

        clientLen = sizeof(sockaddr_in);

        memset(&client, 0, clientLen);

        // client 结构用来接受发包人的网络地址

        ret = recvfrom(server, ReadData,sizeof(ReadData),

                        0, (sockaddr*)(&client), &clientLen);

        if( ret == SOCKET_ERROR )

        {

            if( WSAGetLastError() == WSAEINTR ) //主线程终止时recvfrom返回的错误

                break;

            else

                printf("读取数据失败,错误代码: %d/n",WSAGetLastError());

            break ;

        }

        // 加入结尾字符

        ReadData[ret] = '/0';

        // inet_ntoa 将网络地址转换成点分字符串格式

        printf("收到来至: [ %s ] 的消息: %s",inet_ntoa(client.sin_addr),ReadData);

    }

}

 


 

补充

    IP_MULTICAST_IF 这个选项用于设置自哪个IP接口发出多播数据。正常情况下(即非多播环境),只能参照
路由表来决定一个数据报自哪个接口发出。此时,依据一个特定数据报本身以及它的目的地,系统可判断出哪个接口最适合用来发出这个数据报。然而,由于多播地址可由任何人使用,所以仅仅依靠路由表的自动判断是不够的。程序员必须多少施加一些自己的影响。当然,只有打算发出多播数据的那台机器已通过网卡建立了与多个网络的连接,才需要考虑到这方面的问题。

 

示例:

 

DWORD dwInterface;

 

dwInterface = inet_addr("192.168.2.2");

if (setsockopt(s,IPPROTO_IP,IP_MULTICAST_IF,(char*)&dwInterface,

            sizeof(DWORD)) == SOCKET_ERROR)

{

    // Error

}

 

     上例中,我们将本地接口设为 192.168.2.2  对通过套接字s发出的任何多播数据而言,都会通过分配了那个IP地址的网络接口送出。

这篇关于「多播」WinSock下基于IP协议的多播实现示例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/950604

相关文章

MySQL更新某个字段拼接固定字符串的实现

《MySQL更新某个字段拼接固定字符串的实现》在MySQL中,我们经常需要对数据库中的某个字段进行更新操作,本文就来介绍一下MySQL更新某个字段拼接固定字符串的实现,感兴趣的可以了解一下... 目录1. 查看字段当前值2. 更新字段拼接固定字符串3. 验证更新结果mysql更新某个字段拼接固定字符串 -

java实现延迟/超时/定时问题

《java实现延迟/超时/定时问题》:本文主要介绍java实现延迟/超时/定时问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java实现延迟/超时/定时java 每间隔5秒执行一次,一共执行5次然后结束scheduleAtFixedRate 和 schedu

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio

在Android平台上实现消息推送功能

《在Android平台上实现消息推送功能》随着移动互联网应用的飞速发展,消息推送已成为移动应用中不可或缺的功能,在Android平台上,实现消息推送涉及到服务端的消息发送、客户端的消息接收、通知渠道(... 目录一、项目概述二、相关知识介绍2.1 消息推送的基本原理2.2 Firebase Cloud Me

Spring Boot项目中结合MyBatis实现MySQL的自动主从切换功能

《SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能》:本文主要介绍SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能,本文分步骤给大家介绍的... 目录原理解析1. mysql主从复制(Master-Slave Replication)2. 读写分离3.

Redis实现延迟任务的三种方法详解

《Redis实现延迟任务的三种方法详解》延迟任务(DelayedTask)是指在未来的某个时间点,执行相应的任务,本文为大家整理了三种常见的实现方法,感兴趣的小伙伴可以参考一下... 目录1.前言2.Redis如何实现延迟任务3.代码实现3.1. 过期键通知事件实现3.2. 使用ZSet实现延迟任务3.3

基于Python和MoviePy实现照片管理和视频合成工具

《基于Python和MoviePy实现照片管理和视频合成工具》在这篇博客中,我们将详细剖析一个基于Python的图形界面应用程序,该程序使用wxPython构建用户界面,并结合MoviePy、Pill... 目录引言项目概述代码结构分析1. 导入和依赖2. 主类:PhotoManager初始化方法:__in

springboot filter实现请求响应全链路拦截

《springbootfilter实现请求响应全链路拦截》这篇文章主要为大家详细介绍了SpringBoot如何结合Filter同时拦截请求和响应,从而实现​​日志采集自动化,感兴趣的小伙伴可以跟随小... 目录一、为什么你需要这个过滤器?​​​二、核心实现:一个Filter搞定双向数据流​​​​三、完整代码

SpringBoot利用@Validated注解优雅实现参数校验

《SpringBoot利用@Validated注解优雅实现参数校验》在开发Web应用时,用户输入的合法性校验是保障系统稳定性的基础,​SpringBoot的@Validated注解提供了一种更优雅的解... 目录​一、为什么需要参数校验二、Validated 的核心用法​1. 基础校验2. php分组校验3

Python实现AVIF图片与其他图片格式间的批量转换

《Python实现AVIF图片与其他图片格式间的批量转换》这篇文章主要为大家详细介绍了如何使用Pillow库实现AVIF与其他格式的相互转换,即将AVIF转换为常见的格式,比如JPG或PNG,需要的小... 目录环境配置1.将单个 AVIF 图片转换为 JPG 和 PNG2.批量转换目录下所有 AVIF 图