C/C++ IPV6服务器socket绑定在::,接受ipv4链接(双栈)

2024-05-08 14:12

本文主要是介绍C/C++ IPV6服务器socket绑定在::,接受ipv4链接(双栈),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

先决条件:

1、 创建IPV6套接字

2、打开套接字可重用

3、禁用仅限 IPV6

                BOOL bEnable = FALSE;
                if (setsockopt(listenfd_, IPPROTO_IPV6, IPV6_V6ONLY, reinterpret_cast<char*>(&bEnable), sizeof(bEnable)) < 0)
                {
                    return false;
                }
 

4、绑定套接字

5、使用 struct sockaddr_in6 结构接受套接字,而不是用缺省使用 “sockaddr”,否则 10014(WSAEFAULT)错误,并导致套接字句柄泄漏。

                                struct sockaddr_in6 address = { 0 };
                                int address_size = sizeof(address);

                                int sockfd = WSAAccept(listenfd_, (sockaddr*)&address, &address_size, NULL, NULL);
                                if (sockfd != INVALID_SOCKET)
                                {
                                    AcceptSocketEventArgs e = { Adjust(sockfd) };
                                    OnAcceptSocket(e);
                                }

完整大体代码为;

        bool Open(const char* localIP, int localPort, int backlog) noexcept{if (localPort < IPEndPoint::MinPort || localPort > IPEndPoint::MaxPort){return false;}if (NULL == localIP || *localIP == '\x0'){return false;}if (listenfd_ != INVALID_SOCKET){return false;}if (NULL != hEvent_){return false;}if (NULL != afo_){return false;}if (NULL == context_){return false;}boost::system::error_code ec;boost::asio::ip::address bindIP = StringToAddress(localIP, ec);if (ec){return false;}if (backlog < 1){backlog = PPP_LISTEN_BACKLOG;}in_ = bindIP.is_v4();if (bindIP.is_v6()){struct sockaddr_in6 in6;memset(&in6, 0, sizeof(in6));in6.sin6_family = AF_INET6;in6.sin6_port = htons(localPort);if (inet_pton(AF_INET6, localIP, &in6.sin6_addr) < 1){return false;}listenfd_ = WSASocket(AF_INET6, SOCK_STREAM, IPPROTO_TCP, NULL, 0, WSA_FLAG_OVERLAPPED);if (listenfd_ == INVALID_SOCKET){return false;}Adjust(listenfd_);if (!Socket::ReuseSocketAddress(listenfd_, true)){return false;}BOOL bEnable = FALSE;if (setsockopt(listenfd_, IPPROTO_IPV6, IPV6_V6ONLY, reinterpret_cast<char*>(&bEnable), sizeof(bEnable)) < 0){return false;}if (bind(listenfd_, reinterpret_cast<sockaddr*>(&in6), sizeof(in6)) < 0){return false;}}elif(bindIP.is_v4()){struct sockaddr_in in4;memset(&in4, 0, sizeof(in4));in4.sin_family = AF_INET;in4.sin_port = htons(localPort);if (inet_pton(AF_INET, localIP, &in4.sin_addr) < 1){return false;}listenfd_ = WSASocket(AF_INET, SOCK_STREAM, IPPROTO_TCP, NULL, 0, WSA_FLAG_OVERLAPPED);if (listenfd_ == INVALID_SOCKET){return false;}Adjust(listenfd_);if (!Socket::ReuseSocketAddress(listenfd_, true)){return false;}if (bind(listenfd_, reinterpret_cast<sockaddr*>(&in4), sizeof(in4)) < 0){return false;}}else{return false;}if (listen(listenfd_, backlog) < 0){return false;}hEvent_ = WSACreateEvent();if (hEvent_ == WSA_INVALID_EVENT){return false;}if (WSAEventSelect(listenfd_, hEvent_, FD_ACCEPT | FD_CLOSE) != NOERROR){return false;}afo_ = make_shared_void_pointer<boost::asio::windows::object_handle>(*context_, hEvent_);return Next();}

接受套接字连入部分:

WSANETWORKEVENTS events;
if (WSAEnumNetworkEvents(listenfd, hEvent, &events) == NOERROR)
{if (events.lNetworkEvents & FD_ACCEPT){if (events.iErrorCode[FD_ACCEPT_BIT] == 0){struct sockaddr_in6 address = { 0 };int address_size = sizeof(address);int sockfd = WSAAccept(listenfd_, (sockaddr*)&address, &address_size, NULL, NULL);if (sockfd != INVALID_SOCKET){AcceptSocketEventArgs e = { Adjust(sockfd) };OnAcceptSocket(e);}}}elif(events.lNetworkEvents & FD_CLOSE){if (events.iErrorCode[FD_ACCEPT_BIT] == 0) /* event is operation_canceled. */{return;}}
}

这篇关于C/C++ IPV6服务器socket绑定在::,接受ipv4链接(双栈)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

springboot上传zip包并解压至服务器nginx目录方式

《springboot上传zip包并解压至服务器nginx目录方式》:本文主要介绍springboot上传zip包并解压至服务器nginx目录方式,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录springboot上传zip包并解压至服务器nginx目录1.首先需要引入zip相关jar包2.然

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

将Java项目提交到云服务器的流程步骤

《将Java项目提交到云服务器的流程步骤》所谓将项目提交到云服务器即将你的项目打成一个jar包然后提交到云服务器即可,因此我们需要准备服务器环境为:Linux+JDK+MariDB(MySQL)+Gi... 目录1. 安装 jdk1.1 查看 jdk 版本1.2 下载 jdk2. 安装 mariadb(my

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下: