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

相关文章

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

C++ 右值引用(rvalue references)与移动语义(move semantics)深度解析

《C++右值引用(rvaluereferences)与移动语义(movesemantics)深度解析》文章主要介绍了C++右值引用和移动语义的设计动机、基本概念、实现方式以及在实际编程中的应用,... 目录一、右值引用(rvalue references)与移动语义(move semantics)设计动机1

Nginx服务器部署详细代码实例

《Nginx服务器部署详细代码实例》Nginx是一个高性能的HTTP和反向代理web服务器,同时也提供了IMAP/POP3/SMTP服务,:本文主要介绍Nginx服务器部署的相关资料,文中通过代码... 目录Nginx 服务器SSL/TLS 配置动态脚本反向代理总结Nginx 服务器Nginx是一个‌高性

Prometheus+cpolar如何在手机上也能监控服务器状态?

《Prometheus+cpolar如何在手机上也能监控服务器状态?》本文强调了通过Cpolar这一内网穿透工具,轻松突破Prometheus仅限于局域网访问的限制,实现外网随时随地访问监控数据,教你... 目录前言1.安装prometheus2.安装cpolar实现随时随地开发3.配置公网地址4.保留固定

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias