C语言-多播测试代码(IPv4和IPv6)

简介: C语言-多播测试代码(IPv4和IPv6)

IPv4接收端代码(windows)

#include <stdio.h>  
#include <winsock2.h>  
#include <ws2tcpip.h>  
#pragma comment(lib, "ws2_32.lib") 
#define BUFLEN 255
#define GROUP_ADDR "229.0.0.211"
#define GROUP_PORT 12345
int main(int argc, char **argv)
{
    WSADATA wsaData;
    WSAStartup(MAKEWORD(2, 2), &wsaData);
    struct sockaddr_in peeraddr;
    int sockfd;
    char recmsg[BUFLEN + 1];
    unsigned int socklen;
    struct ip_mreq mreq;
    /* 创建 socket 用于UDP通讯 */
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        printf("socket creating err in udptalk\n");
        exit(1);
    }
    mreq.imr_multiaddr.s_addr = inet_addr(GROUP_ADDR);
    mreq.imr_interface.s_addr = htonl(INADDR_ANY);//本地ip地址
    /* 把本机加入组播地址,即本机网卡作为组播成员,只有加入组才能收到组播消息 */
    if (setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq,
         sizeof(struct ip_mreq)) == -1)
    {
        perror("setsockopt");
        exit(-1);
    }
    socklen = sizeof(struct sockaddr_in);
    memset(&peeraddr, 0, socklen);
    peeraddr.sin_family = AF_INET;
    peeraddr.sin_addr.s_addr  = htonl(INADDR_ANY);
    peeraddr.sin_port = htons(GROUP_PORT);
    /* 绑定自己的端口和IP信息到socket上 */
    if (bind(sockfd, (struct sockaddr *) &peeraddr,
         sizeof(struct sockaddr_in)) == -1)
    {
        printf("Bind error\n");
        exit(0);
    }
    /* 循环接收网络上来的组播消息 */
    for (;;) {
        //bzero(recmsg, BUFLEN + 1);
        memset(recmsg, 0, sizeof(BUFLEN + 1));
        int n = recvfrom(sockfd, recmsg, BUFLEN, 0,
                     (struct sockaddr *) &peeraddr, &socklen);
        if (n < 0) {
            printf("recvfrom err in udptalk!\n");
            exit(4);
        } else {
            /* 成功接收到数据报 */
            recmsg[n] = 0;
            printf("peer:%s\r\n", recmsg);
        }
    }
    closesocket(sockfd);
  WSACleanup();
}

IPv4发送端代码(windows)

#include <stdio.h>  
#include <winsock2.h>  
#include <ws2tcpip.h>  
#pragma comment(lib, "ws2_32.lib") 
int main(int argc, char** argv)
{
  WSADATA wsaData;
  WSAStartup(MAKEWORD(2, 2), &wsaData);
  SOCKET sock = socket(AF_INET, SOCK_DGRAM, 0);
  int n = 0; // 注意, 这个n值很重要, 下面我会讲到  
  setsockopt(sock, IPPROTO_IP, IP_MULTICAST_TTL, (char*)&n, sizeof(n));
  sockaddr_in addr;
  addr.sin_addr.S_un.S_addr = inet_addr("229.0.0.211");//像组播地址发送消息
  addr.sin_family = AF_INET;
  addr.sin_port = htons(12345);
  static int i = 0;
  while (1)
  {
  char buf[100] = { 0 };
  sprintf(buf, "blablablabla:%d", i++);
  sendto(sock, buf, strlen(buf) + 1, 0, (sockaddr*)&addr, sizeof(sockaddr));
  Sleep(500);
  }
  closesocket(sock);
  WSACleanup();
  return 0;
}

IPv6发送端代码(linux)

IPv6多播代码,用于linux下,window下差不多

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#define MULTICAST_ADDR "ff14::2:e" // 多播组地址,这里不能用"ff02::1"等多播组(否则接收端无法接收到数据,原因未知,我草,这个问题找了好久好久,基本上网上好多代码都用ff02::1这个多播组,然后接收方死活收不到组播数据)
//#define MULTICAST_ADDR "ff02::1" 
#define PORT 12345 // Port number
int main() {
    int sockfd;
    struct sockaddr_in6 multicast_addr;
    char message[] = "Hello, multicast11111!";
    // Create socket
    sockfd = socket(AF_INET6, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        perror("socket");
        exit(EXIT_FAILURE);
    }
    // Set socket options for multicast
    int enable = 1;
    if (setsockopt(sockfd, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, &enable, sizeof(enable)) < 0) {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }
    // Set multicast group address and port
    memset(&multicast_addr, 0, sizeof(multicast_addr));
    multicast_addr.sin6_family = AF_INET6;
    multicast_addr.sin6_port = htons(PORT);
    if (inet_pton(AF_INET6, MULTICAST_ADDR, &(multicast_addr.sin6_addr)) <= 0) {
        perror("inet_pton");
        exit(EXIT_FAILURE);
    }
     while(1)
     {
       // Send multicast packet
    if (sendto(sockfd, message, strlen(message), 0, (struct sockaddr*)&multicast_addr,    sizeof(multicast_addr)) < 0) {
        perror("sendto");
        exit(EXIT_FAILURE);
     }
   }
    printf("Multicast packet sent.\n");
    close(sockfd);
    return 0;
}

IPv6接收端代码(linux)

这里列举linux下的组播接收端的代码,windows类似,需要加WSAStartup,另外需要将bind那里不同,linux需要绑定多播地址,而windows需要绑定本机的一个地址,注意看下面bind那里的注释

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <errno.h>
#include <netdb.h>
#define MAX_BUFFER_SIZE 1024
#define MULTICAST_ADDR "ff02::1" 
int main() {
    int sockfd;
    struct addrinfo *multicast_addr = NULL;
    char buffer[MAX_BUFFER_SIZE];
    // Create socket
    sockfd = socket(AF_INET6, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        perror("Failed to create socket");
        exit(EXIT_FAILURE);
    }
    struct addrinfo hint = {0};
    struct addrinfo *info = NULL;
    hint.ai_family = AF_INET6;
    hint.ai_socktype = SOCK_DGRAM;
    int err = getaddrinfo("fe80::d609:e8da:7b4d:7f42",NULL,&hint,&info);//fe80::d609:e8da:7b4d:7f42 是本机的一个本机链接地址
    if(err != 0)
    {
      perror("getaddrinfo");
      exit(1);
    }
    struct sockaddr_in6 *addr = (struct sockaddr_in6*)info->ai_addr;
    inet_pton(AF_INET6,MULTICAST_ADDR,&addr->sin6_addr);//如果是windows,需要将这句注释掉,本机是绑定本机地址,而不是绑定多播地址
   // addr->sin6_addr = in6addr_any;
    addr->sin6_port = htons(12345);
    if(bind(sockfd,(struct sockaddr*)addr,info->ai_addrlen) != 0)
    {
      close(sockfd);
      printf("bind error,errno:%d\n",errno);
      //perror("bind");
      exit(1);
    }
    if(getaddrinfo(MULTICAST_ADDR,NULL,&hint, &multicast_addr) != 0)
    {
      close(sockfd);
      perror("getaddrinfo");
      exit(1);
    }
    struct ipv6_mreq mreq;
    memset( &mreq, 0, sizeof(mreq) );
    memcpy( &mreq.ipv6mr_multiaddr, &((struct sockaddr_in6 *) multicast_addr->ai_addr)->sin6_addr, sizeof(mreq.ipv6mr_multiaddr) );
    mreq.ipv6mr_interface = 0; // 2 happens to be the interface ID; I've tried other values here
    freeaddrinfo(multicast_addr);
   if( setsockopt(sockfd, IPPROTO_IPV6, IPV6_JOIN_GROUP, &mreq, sizeof(mreq) ) != 0 ) {
        close(sockfd);
      perror( "IPV6_JOIN_GROUP" );
        exit( 1 );
   }
   printf("join group success\n");
    // Receive multicast packets
    while (1) {
        ssize_t num_bytes = recvfrom(sockfd, buffer, sizeof(buffer), 0, NULL, 0);
        if (num_bytes < 0) {
            perror("Failed to receive packet");
            exit(EXIT_FAILURE);
        }
        // Process received packet
        // ...
        // Print received data
        printf("Received: %.*s\n", (int)num_bytes, buffer);
    }
    // Close socket
    close(sockfd);
    return 0;
}

参考

https://blog.csdn.net/u010429831/article/details/122495698


相关文章
|
1月前
|
数据采集 机器学习/深度学习 大数据
行为检测代码(一):超详细介绍C3D架构训练+测试步骤
这篇文章详细介绍了C3D架构在行为检测领域的应用,包括训练和测试步骤,使用UCF101数据集进行演示。
40 1
行为检测代码(一):超详细介绍C3D架构训练+测试步骤
|
1月前
|
机器学习/深度学习 人工智能 监控
提升软件质量的关键路径:高效测试策略与实践在软件开发的宇宙中,每一行代码都如同星辰般璀璨,而将这些星辰编织成星系的过程,则依赖于严谨而高效的测试策略。本文将引领读者探索软件测试的奥秘,揭示如何通过精心设计的测试方案,不仅提升软件的性能与稳定性,还能加速产品上市的步伐,最终实现质量与效率的双重飞跃。
在软件工程的浩瀚星海中,测试不仅是发现缺陷的放大镜,更是保障软件质量的坚固防线。本文旨在探讨一种高效且创新的软件测试策略框架,它融合了传统方法的精髓与现代技术的突破,旨在为软件开发团队提供一套系统化、可执行性强的测试指引。我们将从测试规划的起点出发,沿着测试设计、执行、反馈再到持续优化的轨迹,逐步展开论述。每一步都强调实用性与前瞻性相结合,确保测试活动能够紧跟软件开发的步伐,及时适应变化,有效应对各种挑战。
|
2月前
|
Web App开发 JavaScript 前端开发
添加浮动按钮点击滚动到网页底部的纯JavaScript演示代码 IE9、11,Maxthon 1.6.7,Firefox30、31,360极速浏览器7.5.3.308下测试正常
添加浮动按钮点击滚动到网页底部的纯JavaScript演示代码 IE9、11,Maxthon 1.6.7,Firefox30、31,360极速浏览器7.5.3.308下测试正常
|
1月前
|
存储 搜索推荐 C语言
深入C语言指针,使代码更加灵活(二)
深入C语言指针,使代码更加灵活(二)
|
1月前
|
存储 程序员 编译器
深入C语言指针,使代码更加灵活(一)
深入C语言指针,使代码更加灵活(一)
|
1月前
|
C语言
【C语言的完结】:最后的测试题
【C语言的完结】:最后的测试题
19 3
|
1月前
|
C语言
深入C语言指针,使代码更加灵活(三)
深入C语言指针,使代码更加灵活(三)
深入C语言指针,使代码更加灵活(三)
|
2月前
|
SQL JavaScript 前端开发
基于Python访问Hive的pytest测试代码实现
根据《用Java、Python来开发Hive应用》一文,建立了使用Python、来开发Hive应用的方法,产生的代码如下
69 6
基于Python访问Hive的pytest测试代码实现
|
2月前
|
Java C++
代码文件间重复性测试
本文介绍了如何使用代码相似性检测工具simian来找出代码文件中的重复行,并通过示例指令展示了如何将检测结果输出到指定的文本文件中。
|
2月前
|
安全 C语言
在C语言中,正确使用运算符能提升代码的可读性和效率
在C语言中,运算符的使用需要注意优先级、结合性、自增自减的形式、逻辑运算的短路特性、位运算的类型、条件运算的可读性、类型转换以及使用括号来明确运算顺序。掌握这些注意事项可以帮助编写出更安全和高效的代码。
49 4