C语言 网络编程(十六)广播和组播

简介: 广播和组播是网络通信的重要方式。广播允许一台主机向子网内所有主机发送数据包,常用于局域网内的消息传播;组播则将数据包发送给特定的一组主机,适用于视频会议等应用场景。广播地址如 `192.168.1.255` 用于同一子网的所有主机。组播地址如 `224.0.0.0` 至 `239.255.255.255` 标识特定主机群。C语言示例展示了如何通过 UDP 实现广播和组播通信。此外,UNIX域套接字用于同一机器上进程间的高效通信。

网络(十六)广播和组播

广播

广播(Broadcast)是一种网络通信方式,允许一台主机向网络中的所有其他主机发送数据包。

在IP网络中,广播地址用于向同一子网中的所有主机发送数据包。

广播地址通常是子网中的最后一个地址,例如,如果子网掩码是255.255.255.0,那么广播地址就是192.168.1.255。

只有用户数据报(使用UDP协议)套接字才能广播

数据包发送方式只有一个接受方,称为单播

如果同时发给局域网中的所有主机,称为广播

如果同时发给局域网中的部分主机,称为组播

在局域网中,广播是一种将消息发送给所有主机的通信方式。以下是广播在局域网中发送的基本原理和步骤:

  1. 广播地址:
    在以太网中,广播地址是一个特殊的MAC地址,通常是全FF(即FF:FF:FF:FF:FF:FF)。
    在IP网络中,广播地址通常是网络地址加上全1的子网掩码。例如,在一个C类网络中,如果网络地址是192.168.1.0,子网掩码是255.255.255.0,那么广播地址就是192.168.1.255。
  2. 数据包发送:
    当一台主机需要发送广播消息时,它会将要发送的数据包的目的MAC地址设置为广播地址(FF:FF:FF:FF:FF:FF)。
    数据包的源MAC地址设置为发送主机的MAC地址。
  3. 数据包传输:
    交换机或集线器接收到这个数据包后,会将其转发给连接到它的所有其他端口。
    这样,局域网中的所有主机都会收到这个广播数据包。
  4. 数据包处理:
    每台主机在接收到广播数据包后,会检查数据包的目的MAC地址。
    如果目的MAC地址是广播地址,主机会处理这个数据包。
  5. 应用层广播:
    在应用层,例如在UDP协议中,应用程序可以发送一个目的IP地址为广播地址的数据包。
    操作系统会将这个数据包封装成一个以太网帧,并将目的MAC地址设置为广播地址,然后发送出去。

在C语言中,使用广播进行通信通常涉及以下步骤:

  1. 创建套接字:使用 socket() 函数创建一个UDP套接字。
  2. 设置广播选项:使用 setsockopt() 函数设置套接字的广播选项,允许套接字发送广播数据包。
  3. 设置广播地址:使用 sockaddr_in 结构体设置广播地址和端口。
  4. 发送数据:使用 sendto() 函数发送广播数据包。

示例

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

// 定义广播端口和广播IP地址
#define PORT 8080
#define BROADCAST_IP "255.255.255.255"
#define BUFSIZE 1024

int main() {
   
    int sockfd; // 套接字文件描述符
    struct sockaddr_in broadcast_addr; // 广播地址结构体
    char *message = "Hello, this is a broadcast message!"; // 要发送的广播消息
    int broadcastEnable = 1; // 广播选项,设置为1表示启用广播

    // 创建UDP套接字
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
   
        perror("socket failed"); // 如果创建失败,输出错误信息
        exit(EXIT_FAILURE); // 退出程序
    }

    // 设置广播选项
    if (setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, &broadcastEnable, sizeof(broadcastEnable)) < 0) {
   
        perror("setsockopt failed"); // 如果设置失败,输出错误信息
        close(sockfd); // 关闭套接字
        exit(EXIT_FAILURE); // 退出程序
    }

    // 设置广播地址
    memset(&broadcast_addr, 0, sizeof(broadcast_addr)); // 初始化广播地址结构体
    broadcast_addr.sin_family = AF_INET; // 设置地址族为IPv4
    broadcast_addr.sin_addr.s_addr = inet_addr(BROADCAST_IP); // 设置广播IP地址
    broadcast_addr.sin_port = htons(PORT); // 设置广播端口

    // 发送广播数据
    if (sendto(sockfd, message, strlen(message), 0, (struct sockaddr *)&broadcast_addr, sizeof(broadcast_addr)) < 0) {
   
        perror("sendto failed"); // 如果发送失败,输出错误信息
        close(sockfd); // 关闭套接字
        exit(EXIT_FAILURE); // 退出程序
    }

    printf("Broadcast message sent successfully.\n"); // 发送成功,输出提示信息

    // 关闭套接字
    close(sockfd);
    return 0;
}

组播

组播(Multicast)是一种网络通信方式,它允许将数据包发送给一组特定的主机,而不是所有主机。组播通过使用特定的组播地址来实现,这些地址标识了一组希望接收相同数据的主机。
以下是组播的基本概念和原理:




1.组播地址:

组播地址是一个特殊的IP地址范围,用于标识一组主机。
在IPv4中,组播地址的范围是224.0.0.0到239.255.255.255。
在IPv6中,组播地址以FF开头,后面跟着一个标识符和组ID。

2.组播组:

主机可以加入一个或多个组播组,通过加入组播组,主机表明它希望接收发送到该组播地址的数据包。
组播组是动态的,主机可以随时加入或离开组播组。

3.数据包发送:

当一台主机需要发送组播数据包时,它会将要发送的数据包的目的IP地址设置为组播地址。
数据包的源IP地址设置为发送主机的IP地址。

4.数据包传输:

路由器和交换机在接收到组播数据包后,会根据组播路由协议(如IGMP、PIM等)将数据包转发给所有加入该组播组的主机。
这样,只有那些加入该组播组的主机才会收到这个数据包。

5.应用层组播:

在应用层,例如在UDP协议中,应用程序可以发送一个目的IP地址为组播地址的数据包。
操作系统会将这个数据包封装成一个以太网帧,并将目的MAC地址设置为对应的组播MAC地址,然后发送出去。
组播的优点在于它能够有效地将数据传输给一组特定的主机,而不需要发送多个单播数据包,从而节省了网络带宽。组播广泛应用于视频会议、在线直播、软件更新等需要同时向多个接收者传输数据的场景。

在C语言中,使用广播进行通信通常涉及以下步骤:

  1. 创建套接字:使用 socket() 函数创建一个UDP套接字。
  2. 设置组播选项:使用 setsockopt() 函数设置套接字的组播选项,允许套接字发送组播数据包。
  3. 设置组播地址:使用 sockaddr_in 结构体设置组播地址和端口。
  4. 加入组播组:使用 setsockopt() 函数加入组播组。
  5. 发送数据:使用 sendto() 函数发送组播数据包。

示例

设置本地地址加入多播组

使用setsockopt函数设置 参数是一个 struct ip_mreqn 结构体
struct ip_mreqn {
   
    struct in_addr imr_multiaddr; /* 多播组的IP地址 */
    struct in_addr imr_address;   /* 本地IP地址 */
    int            imr_ifindex;   /* 接口索引 填0 即可 */
};

struct ip_mreqn mreqn;
mreqn.imr_multiaddr.s_addr = inet_addr("多播组的IP地址");
mreqn.imr_address.s_addr = inet_addr("本地IP地址");
mreqn.imr_ifindex = 0;
if(-1 == setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreqn, sizeof(mreqn)))
    ERR_LOG("setsockopt error");

接收端

#include <my_head.h>

int main(int argc, const char *argv[])
{
   
    //创建套接字
    int sockfd = 0;
    if(-1 == (sockfd = socket(AF_INET, SOCK_DGRAM, 0)))
        ERR_LOG("socket error");

    //填充服务器网络信息结构体
    struct sockaddr_in serveraddr;
    serveraddr.sin_family = AF_INET;
    serveraddr.sin_port = htons(atoi(argv[2]));
    //要填组播的IP地址  224-239
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
    socklen_t serveraddrlen = sizeof(serveraddr);

    //绑定
    if(-1 == bind(sockfd, (struct sockaddr *)&serveraddr, serveraddrlen))
        ERR_LOG("bind error");

    //设置加入多播组
    struct ip_mreqn mreqn;
    mreqn.imr_multiaddr.s_addr = inet_addr(argv[1]);
    mreqn.imr_address.s_addr = inet_addr("192.168.2.91");
    mreqn.imr_ifindex = 0;
    if(-1 == setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreqn, sizeof(mreqn)))
        ERR_LOG("setsockopt error");

    //定义结构体保存发送方信息
    struct sockaddr_in clientaddr;
    socklen_t clientaddrlen = sizeof(clientaddr);

    //接收数据
    char buff[128];
    while(1){
   
        memset(buff, 0, sizeof(buff));
        if(-1 == recvfrom(sockfd, buff, sizeof(buff), 0, (struct sockaddr *)&clientaddr, &clientaddrlen))
            ERR_LOG("recvfrom error");
        printf("客户端[%s:%d]发来数据[%s]\n", inet_ntoa(clientaddr.sin_addr), ntohs(clientaddr.sin_port), buff);
    }

    //关闭套接字
    close(sockfd);
    return 0;
}

发送端

#include <my_head.h>

int main(int argc, const char *argv[])
{
   
    //创建套接字
    int sockfd = 0;
    if(-1 == (sockfd = socket(AF_INET, SOCK_DGRAM, 0)))
        ERR_LOG("socket error");

    //填充服务器网络信息结构体
    struct sockaddr_in serveraddr;
    serveraddr.sin_family = AF_INET;
    serveraddr.sin_port = htons(atoi(argv[2]));
    //要填组播的IP地址  224-239
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
    socklen_t serveraddrlen = sizeof(serveraddr);

    //发送数据
    char buff[128];
    while(1){
   
        fgets(buff, sizeof(buff), stdin);
        buff[strlen(buff)-1] = '\0';
        if(-1 == sendto(sockfd, buff, sizeof(buff), 0, (struct sockaddr *)&serveraddr, serveraddrlen))
            ERR_LOG("sendto error");
    }

    //关闭套接字
    close(sockfd);
    return 0;
}

本地通信(UNIX域套接字)

在本地通信中,UNIX域套接字是一种高效的进程间通信(IPC)机制。它允许同一台机器上的不同进程之间进行通信。

UNIX域套接字是一种基于文件系统的IPC机制,它不需要网络协议栈的支持。

UNIX域套接字主要使用以下几个系统调用函数:

    socket():创建一个新的套接字。
    bind():将套接字绑定到一个地址。
    listen():使套接字进入监听状态,等待客户端连接。
    accept():接受一个客户端连接。
    connect():连接到服务器端的套接字。
    send()recv():发送和接收数据。
    close():关闭套接字。
服务器端:

    socket():创建一个UNIX域套接字。
    bind():将套接字绑定到一个路径(例如 /tmp/my_unix_socket)。
    listen():使套接字进入监听状态,等待客户端连接。
    accept():接受一个客户端连接。
    recv():接收数据。
    send():发送响应。
    close():关闭套接字。
    unlink():删除套接字文件。


客户端:

    socket():创建一个UNIX域套接字。
    connect():连接到服务器端的套接字路径。
    send():发送数据。
    recv():接收响应。
    close():关闭套接字。

服务器端代码

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

#define SOCKET_PATH "/tmp/my_unix_socket"

int main() {
   
    struct sockaddr_un server_addr;
    int server_socket, client_socket;
    socklen_t client_len;
    char buffer[1024];

    // 创建UNIX域套接字
    if ((server_socket = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) {
   
        perror("socket");
        exit(EXIT_FAILURE);
    }

    // 设置服务器地址
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sun_family = AF_UNIX;
    strncpy(server_addr.sun_path, SOCKET_PATH, sizeof(server_addr.sun_path) - 1);

    // 绑定套接字
    if (bind(server_socket, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) {
   
        perror("bind");
        close(server_socket);
        exit(EXIT_FAILURE);
    }

    // 监听连接
    if (listen(server_socket, 5) == -1) {
   
        perror("listen");
        close(server_socket);
        exit(EXIT_FAILURE);
    }

    printf("服务器正在监听...\n");

    while (1) {
   
        // 接受连接
        client_len = sizeof(struct sockaddr_un);
        if ((client_socket = accept(server_socket, (struct sockaddr *)&server_addr, &client_len)) == -1) {
   
            perror("accept");
            continue;
        }

        printf("连接已建立\n");

        // 接收数据
        ssize_t num_bytes = recv(client_socket, buffer, sizeof(buffer), 0);
        if (num_bytes > 0) {
   
            buffer[num_bytes] = '\0';
            printf("收到数据: %s\n", buffer);

            // 发送响应
            send(client_socket, "Hello from server", 17, 0);
        }

        close(client_socket);
    }

    close(server_socket);
    unlink(SOCKET_PATH);
    return 0;
}

客户端代码

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

#define SOCKET_PATH "/tmp/my_unix_socket"

int main() {
   
    struct sockaddr_un server_addr;
    int client_socket;
    char buffer[1024];

    // 创建UNIX域套接字
    if ((client_socket = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) {
   
        perror("socket");
        exit(EXIT_FAILURE);
    }

    // 设置服务器地址
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sun_family = AF_UNIX;
    strncpy(server_addr.sun_path, SOCKET_PATH, sizeof(server_addr.sun_path) - 1);

    // 连接到服务器
    if (connect(client_socket, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) {
   
        perror("connect");
        close(client_socket);
        exit(EXIT_FAILURE);
    }

    printf("已连接到服务器\n");

    // 发送数据
    send(client_socket, "Hello from client", 17, 0);

    // 接收响应
    ssize_t num_bytes = recv(client_socket, buffer, sizeof(buffer), 0);
    if (num_bytes > 0) {
   
        buffer[num_bytes] = '\0';
        printf("收到响应: %s\n", buffer);
    }

    close(client_socket);
    return 0;
}
相关文章
|
14天前
|
网络协议 算法 网络性能优化
C语言 网络编程(十五)套接字选项设置
`setsockopt()`函数用于设置套接字选项,如重复使用地址(`SO_REUSEADDR`)、端口(`SO_REUSEPORT`)及超时时间(`SO_RCVTIMEO`)。其参数包括套接字描述符、协议级别、选项名称、选项值及其长度。成功返回0,失败返回-1并设置`errno`。示例展示了如何创建TCP服务器并设置相关选项。配套的`getsockopt()`函数用于获取这些选项的值。
|
14天前
|
网络协议 C语言
C语言 网络编程(十三)并发的TCP服务端-以进程完成功能
这段代码实现了一个基于TCP协议的多进程并发服务端和客户端程序。服务端通过创建子进程来处理多个客户端连接,解决了粘包问题,并支持不定长数据传输。客户端则循环发送数据并接收服务端回传的信息,同样处理了粘包问题。程序通过自定义的数据长度前缀确保了数据的完整性和准确性。
|
14天前
|
网络协议 C语言
C语言 网络编程(十一)TCP通信创建流程---服务端
在服务器流程中,新增了绑定IP地址与端口号、建立监听队列及接受连接并创建新文件描述符等步骤。`bind`函数用于绑定IP地址与端口,`listen`函数建立监听队列并设置监听状态,`accept`函数则接受连接请求并创建新的文件描述符用于数据传输。套接字状态包括关闭(CLOSED)、同步发送(SYN-SENT)、同步接收(SYN-RECEIVE)和已建立连接(ESTABLISHED)。示例代码展示了TCP服务端程序如何初始化socket、绑定地址、监听连接请求以及接收和发送数据。
|
14天前
|
网络协议 C语言
C语言 网络编程(十四)并发的TCP服务端-以线程完成功能
这段代码实现了一个基于TCP协议的多线程服务器和客户端程序,服务器端通过为每个客户端创建独立的线程来处理并发请求,解决了粘包问题并支持不定长数据传输。服务器监听在IP地址`172.17.140.183`的`8080`端口上,接收客户端发来的数据,并将接收到的消息添加“-回传”后返回给客户端。客户端则可以循环输入并发送数据,同时接收服务器回传的信息。当输入“exit”时,客户端会结束与服务器的通信并关闭连接。
|
14天前
|
C语言
C语言 网络编程(八)并发的UDP服务端 以进程完成功能
这段代码展示了如何使用多进程处理 UDP 客户端和服务端通信。客户端通过发送登录请求与服务端建立连接,并与服务端新建的子进程进行数据交换。服务端则负责接收请求,验证登录信息,并创建子进程处理客户端的具体请求。子进程会创建一个新的套接字与客户端通信,实现数据收发功能。此方案有效利用了多进程的优势,提高了系统的并发处理能力。
|
14天前
|
网络协议 C语言
C语言 网络编程(十二)TCP通信创建-粘包
TCP通信中的“粘包”现象指的是由于协议特性,发送方的数据包被拆分并在接收方按序组装,导致多个数据包粘连或单个数据包分割。为避免粘包,可采用定长数据包或先传送数据长度再传送数据的方式。示例代码展示了通过在发送前添加数据长度信息,并在接收时先读取长度后读取数据的具体实现方法。此方案适用于长度不固定的数据传输场景。
|
14天前
|
网络协议 C语言
C语言 网络编程(十)TCP通信创建流程---客户端
在TCP通信中,客户端需通过一系列步骤与服务器建立连接并进行数据传输。首先使用 `socket()` 函数创建一个流式套接字,然后通过 `connect()` 函数连接服务器。连接成功后,可以使用 `send()` 和 `recv()` 函数进行数据发送和接收。最后展示了一个完整的客户端示例代码,实现了与服务器的通信过程。
|
14天前
|
C语言
C语言 网络编程(九)并发的UDP服务端 以线程完成功能
这是一个基于UDP协议的客户端和服务端程序,其中服务端采用多线程并发处理客户端请求。客户端通过UDP向服务端发送登录请求,并根据登录结果与服务端的新子线程进行后续交互。服务端在主线程中接收客户端请求并创建新线程处理登录验证及后续通信,子线程创建新的套接字并与客户端进行数据交换。该程序展示了如何利用线程和UDP实现简单的并发服务器架构。
|
3天前
|
SQL 安全 网络安全
网络安全与信息安全:构建防线的三大支柱在数字时代,网络安全和信息安全成为了我们不可忽视的重要议题。本文将深入探讨网络安全漏洞、加密技术以及安全意识这三大支柱,帮助您建立更全面的安全防护体系。
本文旨在分享有关网络安全漏洞、加密技术和安全意识的知识。首先,我们将介绍常见的网络安全漏洞及其形成原因;接着,我们将探讨几种主要的加密技术及其应用;最后,我们将强调提高安全意识的重要性并提供实用的建议。通过这些内容,读者可以更好地理解如何在日常生活和工作中保护自己的信息安全。
23 9
|
2天前
|
存储 SQL 安全
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享
【9月更文挑战第15天】在数字化时代,网络安全与信息安全已成为我们日常生活中不可或缺的一部分。本文将介绍网络安全漏洞、加密技术和安全意识等方面的知识,以帮助读者更好地了解和应对网络安全挑战。通过深入浅出的方式,我们将探讨如何保护个人信息和数据安全,以及如何提高自己的网络安全意识。