当前位置: 首页 > 面试题库 >

如何在Linux上的C / C ++中使用ipv6 udp套接字进行多播?

吕俊美
2023-03-14
问题内容

(英语不是我的母语,不用担心某些句子是否很奇怪;)。

我正在开发 PONG游戏, 并且通过创建一些类来帮助我管理窗口,事件…和 网络, 因为我在游戏中添加了 LAN功能
,但是当前您必须输入您想与之联系的人的地址。一起玩。解决此问题的方法是 广播(扫描播放器的LAN)
。对于ipv4来说这很容易,只需使用地址255.255.255.255,但是我们在2017年就提供了仅适用于ipv4的功能。

然后,我寻找一种使用ipv6进行广播的方法,并且了解了多播,但是这部分让我迷失了方向。=(

在Linux上使用C ++的标准库中
,发现了几个不适用于我的多播示例。我目前所做的最好的工作是将udp数据包从该程序的一个实例发送到同一台计算机上的另一个实例。

如何在C / C ++中的Linux上使用ipv6 udp套接字进行多播?

在Internet上找到的最佳代码(我对其进行了重新排列)几乎可以正常工作(客户端和服务全部合而为一,可以通过在argv上加上1或0来选择):

int main(int argc, char const *argv[]) {

struct sockaddr_in6 groupSock;
int sd = -1;

char databuf[10];
int datalen = sizeof databuf;

/* Create a datagram socket on which to send/receive. */
if((sd = socket(AF_INET6, SOCK_DGRAM, 0)) < 0) {
    perror("Opening datagram socket error");
    return 1;
} else {
    cout << "Opening the datagram socket...OK." << endl;;
}

/* Enable SO_REUSEADDR to allow multiple instances of this */
/* application to receive copies of the multicast datagrams. */
int reuse = 1;
if(setsockopt(sd, SOL_SOCKET, SO_REUSEADDR, (char *)&reuse, sizeof reuse) < 0) {
    perror("Setting SO_REUSEADDR error");
    close(sd);
    return 1;
} else {
    cout << "Setting SO_REUSEADDR...OK." << endl;
}

/* Initialize the group sockaddr structure with a */
memset((char *) &groupSock, 0, sizeof groupSock);
groupSock.sin6_family = AF_INET6;
// address of the group
inet_pton(AF_INET6, "ff0e::/16", &groupSock.sin6_addr);
groupSock.sin6_port = htons(4321);

/* Set local interface for outbound multicast datagrams. */
/* The IP address specified must be associated with a local, */
/* multicast capable interface. */
int ifindex = if_nametoindex ("enp3s0");
cout << "ifindex is " << ifindex << endl;

if(setsockopt(sd, IPPROTO_IPV6, IPV6_MULTICAST_IF, &ifindex, sizeof ifindex)) {
    perror("Setting local interface error");
    return 1;
} else {
    cout << "Setting the local interface...OK" << endl;
}

// choice is 0 for sending and 1 for receiving
int choice;
if (argc < 2) {
    cout << "missing argv[1]" << endl;
    return 1;
}
sscanf (argv[1], "%d", &choice);

// if sending
if (choice == 0) {
    memset(databuf, 'a', datalen);
    databuf[sizeof databuf - 1] = '\0';

    if (sendto(sd, databuf, datalen, 0, (sockaddr*)&groupSock, sizeof groupSock) < 0) {
        cout << "Error in send" << endl;
    } else {
        cout << "Send okay!" << endl;
    }
}

// if receiving
else if (choice == 1) {
    groupSock.sin6_addr = in6addr_any;
    if(bind(sd, (sockaddr*)&groupSock, sizeof groupSock)) {
        perror("Binding datagram socket error");
        close(sd);
        return 1;
    } else {
        cout << "Binding datagram socket...OK." << endl;
    }

    /* Join the multicast group ff0e::/16 on the local  */
    /* interface. Note that this IP_ADD_MEMBERSHIP option must be */
    /* called for each local interface over which the multicast */
    /* datagrams are to be received. */
    struct ipv6_mreq group;
    inet_pton (AF_INET6, "ff0e::", &group.ipv6mr_multiaddr.s6_addr);
    group.ipv6mr_interface = ifindex;

    if(setsockopt(sd, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, (char *)&group, sizeof group) < 0) {
        perror("Adding multicast group error");
        close(sd);
        return 1;
    } else {
        cout << "Adding multicast group...OK." << endl;
    }

    if (read(sd, databuf, datalen) < 0) {
        perror("Error in read");
    } else {
        databuf[sizeof databuf - 1] = '\0';// just for safety
        cout << "Read Okay" << endl;
        cout << "Message is : " << databuf << endl;
    }
}

return 0;
}

这里的地址是ff0e ::,但是我尝试使用ff01 ::和ff02 ::。

我需要帮助,但还没有找到任何简单的文档。预先感谢您的任何答复。

编辑:感谢Ron Maupin和Jeremy Friesner的这些评论,它对我有帮助。

编辑:谢谢杰里米!您的建议是使用ff12 :: blah:blah(…)代替ff0e ::可以!我应该为问题写答案以关闭线程吗?


问题答案:

下面的代码是正确的:
唯一错误的是用于多播的地址。
就像Je​​remy所说的那样, ff0e ::不正确,我改用ff12 :: feed:a:dead:beef可以正常工作

使用if_nameindex()可以获得可用接口的名称和索引。

更新:我试图删除一些代码,看看没有它是否可以工作,我设法做到了:

服务器:

// OPEN
int fd = socket(AF_INET6, SOCK_DGRAM, 0);

// BIND
struct sockaddr_in6 address = {AF_INET6, htons(4321)};
bind(fd, (struct sockaddr*)&address, sizeof address);

// JOIN MEMBERSHIP
struct ipv6_mreq group;
group.ipv6mr_interface = 0;
inet_pton(AF_INET6, "ff12::1234", &group.ipv6mr_multiaddr);
setsockopt(fd, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, &group, sizeof group);

// READ
char buffer[128];
read(fd, buffer, sizeof buffer);

客户:

// OPEN
int fd = socket(AF_INET6, SOCK_DGRAM, 0);

// ADDRESS
struct sockaddr_in6 address = {AF_INET6, htons(4321)};
inet_pton(AF_INET6, "ff12::1234", &address.sin6_addr);

// SEND TO
char buffer[128];
strcpy(buffer, "hello world!");
sendto(fd, buffer, sizeof buffer, 0, (struct sockaddr*)&address, sizeof address);


 类似资料:
  • 问题内容: 有没有办法在用户空间或内核中使用C获取Linux中所有打开的套接字的列表(套接字地址或套接字描述符)? 谢谢 问题答案: 打开并阅读以下内容: -打开的TCP套接字列表 -打开的UDP套接字列表 -列出所有“原始”套接字 这些就像您使用文件句柄打开并读取的“常规”文件一样,它将为您提供所有可能需要的有关每个套接字的信息。

  • 问题内容: 如何使用ac api(或与此相关的任何其他语言)获得与命令相似的功能?我需要的不仅仅是信息;特别是与SMTP相关的数据。 问题答案: 如果阻塞(同步)查询没问题,只需使用,然后将程序与- lresolv链接即可。

  • 我必须通过udp套接字发送多个变量类型:int数组和char。我想用相同的udp数据包发送它。标准溶液是什么?将所有内容转换为字节或类似的内容? 我使用的是:sendto(int-sockfd,const-void*buf,size\t len,int-flags,const-struct-sockaddr*dest\u-addr,socklen\t-addrlen); 我的代码是这样的:

  • 问题内容: 我想使用能够发送和接收文件的套接字和C / C ++语言实现在Linux上运行的客户端-服务器体系结构。有没有可以简化此任务的库?有人可以举个例子吗? 问题答案: 最可移植的解决方案是读取文件,然后以循环方式将数据写到套接字中(同样,接收文件时也采用另一种方法)。您可以在该缓冲区中分配一个缓冲区,并从该缓冲区中将其分配到您的套接字中(您也可以使用和,这是套接字特定的写入和读取数据的方式

  • 问题内容: 我是Visual Studio用户,习惯于调试断点。我现在在Linux环境中工作,并且正在使用Eclipse作为IDE。我是linux和eclipse的新手。我不知道如何在Eclipse中使用gdb。我尝试在命令行中使用gdb,但它并不像拥有UI那样容易。 如何在Eclipse中使用gdb? 问题答案: 以下说明适用于Eclipse 3.5(Galileo)。对于3.6(Helios)

  • 我想在服务器端和客户端使用多播套接字。服务器将发送数据,客户端将接收数据。同时,如果数据丢失,客户端将向服务器发送nack数据包,服务器将收到nack数据包。我应该如何实现这一点,一个多播组和一个套接字端口,两个多播组和两个套接字端口?我希望服务器可以同时发送数据和接收nack,这可能吗? 非常感谢。