C++ IPv4与IPv6的兼容编码

这里不再对IPv6 socket相关编程的基础知识进行讲解,只提供一个IP协议无关的服务端和客户端的代码,仅供参考。

服务端代码:

#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <errno.h>
#include <time.h>

using namespace std;

int tcp_listen(const char *host, const char *service, const int listen_num = 5)
{
	int listenfd, ret;
	const int on = 1;
	struct addrinfo hints, *res, *ressave;
	bzero(&hints, sizeof(hints));
	hints.ai_flags = AI_PASSIVE;
	hints.ai_family = AF_UNSPEC;
	hints.ai_socktype = SOCK_STREAM;
	hints.ai_protocol = IPPROTO_IP;

	if (0 != (ret = getaddrinfo(host, service, &hints, &res)))
	{
		cout << "getaddrinfo error: " << gai_strerror(ret) << endl;
		return -1;
	}

	ressave = res;
	while(NULL != res)
	{
		if (-1 == (listenfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)))
		{
			cout << "create socket error: " << strerror(errno) << endl;
			res = res->ai_next;
			continue;
		}

		if (-1 == setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)))
		{
			cout << "setsockopt error: " << strerror(errno) << endl;
			close(listenfd);
			res = res->ai_next;
			continue;
		}

		if (-1 == bind(listenfd, res->ai_addr, res->ai_addrlen))
		{
			cout << "bind error: " << strerror(errno) << endl;
                        close(listenfd);
                        res = res->ai_next;
                        continue;
		}

		if (-1 == listen(listenfd, listen_num))
		{
			cout << "listen error: " << strerror(errno) << endl;
                        close(listenfd);
                        res = res->ai_next;
                        continue;
		}

		break;
	}

	freeaddrinfo(ressave);

	if (NULL == res)
		return -1;

	return listenfd;
}

int get_addrinfo(const struct sockaddr *addr, string &ip, in_port_t &port)
{
	void *numeric_addr = NULL;
	char addr_buff[INET6_ADDRSTRLEN];

	if (AF_INET == addr->sa_family)
	{
		numeric_addr = &((struct sockaddr_in*)addr)->sin_addr;
		port = ntohs(((struct sockaddr_in*)addr)->sin_port);
	}
	else if (AF_INET6 == addr->sa_family)
	{
		numeric_addr = &((struct sockaddr_in6*)addr)->sin6_addr;
		port = ntohs(((struct sockaddr_in6*)addr)->sin6_port);
	}
	else
	{
		return -1;
	}

	if (NULL != inet_ntop(addr->sa_family, numeric_addr, addr_buff, sizeof(addr_buff)))
		ip = addr_buff;
	else
		return -1;

	return 0;
}

int main(int argc, char *argv[])
{
	int listenfd, connfd;
	struct sockaddr_storage cliaddr;
	socklen_t len = sizeof(cliaddr);
	time_t now;
	char buff[128];

	if (2 == argc) //指定端口
		listenfd = tcp_listen(NULL, argv[1]);
	else if (3 == argc) //指定本地IP和端口
		listenfd = tcp_listen(argv[1], argv[2]);
	else
	{
                cout << "usage: " << argv[0] << " [<hostname/ipaddress>] <service/port>" << endl;
		return -1;
	}

	if (listenfd < 0)
	{
		cout << "call tcp_listen error" << endl;
		return -1;
	}

	while (true)
	{
		connfd = accept(listenfd, (struct sockaddr*)&cliaddr, &len);

		string ip = "";
		in_port_t port = 0;
		get_addrinfo((struct sockaddr*)&cliaddr, ip, port);
		cout << "client " << ip << "|" << port << " login" << endl;

		now = time(NULL);
		snprintf(buff, sizeof(buff) - 1, "%.24s", ctime(&now));
		write(connfd, buff, strlen(buff));
		close(connfd);
	}

	close(listenfd);
	return 0;
}


客户端代码:

#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <errno.h>
#include <time.h>

using namespace std;

int tcp_connect(const char *host, const char *service)
{
	int sockfd, ret;
	struct addrinfo hints, *res, *ressave;
	bzero(&hints, sizeof(hints));
	hints.ai_family = AF_UNSPEC;
	hints.ai_socktype = SOCK_STREAM;
	hints.ai_protocol = IPPROTO_IP;

	if (0 != (ret = getaddrinfo(host, service, &hints, &res)))
	{
		cout << "getaddrinfo error: " << gai_strerror(ret) << endl;
		return -1;
	}

	ressave = res;
	while (NULL != res)
	{
		if (-1 == (sockfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)))
                {
                        cout << "create socket error: " << strerror(errno) << endl;
                        res = res->ai_next;
                        continue;
                }

		if (-1 == connect(sockfd, res->ai_addr, res->ai_addrlen))
		{
			cout << "connect error: " << strerror(errno) << endl;
                        close(sockfd);
			res = res->ai_next;
                        continue;
		}

		break;
	}
	
	freeaddrinfo(ressave);

	if (NULL == res)
		return -1;

	return sockfd;
}

int main(int argc, char *argv[])
{
	int sockfd, n;
	char buff[128];
	struct sockaddr_storage cliaddr;
	
	if (3 != argc)
	{
		cout << "usage: " << argv[0] << " <hostname/ipaddress> <service/port>" << endl;
		return -1;
	}

	sockfd = tcp_connect(argv[1], argv[2]);
	if (sockfd < 0)
	{
		cout << "call tcp_connect error" << endl;
		return -1;
	}

	bzero(buff, sizeof(buff));
	while ((n = read(sockfd, buff, sizeof(buff) - 1) > 0))
	{
		cout << buff << endl;
		bzero(buff, sizeof(buff));
	}
	close(sockfd);

	return 0;
}


编译:

g++ daytimesrv.cpp -o daytimesrv

g++ daytimecli.cpp -o daytimecli


运行举例:

服务端运行情况:



客户端运行情况:



以上客户端的请求与服务端的输出一一对应,客户端可以尝试用不同的IP去连接,然后观察服务端的输出,有助于理解底层的交互过程。

  • 2
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
以下是一个使用gsoap实现同时支持ipv4ipv6的设备发现的C++代码示例: ```c++ #include "soapH.h" #include "soapStub.h" #include <iostream> #include <cstring> #include <netdb.h> #include <arpa/inet.h> using namespace std; int main(int argc, char *argv[]) { struct soap soap; soap_init(&soap); // 设置设备发现的多播地址 const char *multicast_addr = "239.255.255.250"; // ipv4多播地址 const char *multicast_addr_v6 = "ff02::c"; // ipv6多播地址 const int multicast_port = 1900; // 设备发现使用的端口号 // 创建socket int sockfd = socket(AF_INET6, SOCK_DGRAM, IPPROTO_UDP); if (sockfd < 0) { perror("socket"); return -1; } // 设置IPV6_MULTICAST_IF选项,指定通过哪个网络接口来发送多播数据包 struct sockaddr_in6 sin6; memset(&sin6, 0, sizeof(sin6)); sin6.sin6_family = AF_INET6; sin6.sin6_port = htons(multicast_port); sin6.sin6_addr = in6addr_any; if (setsockopt(sockfd, IPPROTO_IPV6, IPV6_MULTICAST_IF, &sin6, sizeof(sin6)) < 0) { perror("setsockopt"); close(sockfd); return -1; } // 设置IPV6_V6ONLY选项,表示socket同时支持ipv4ipv6 int opt = 0; if (setsockopt(sockfd, IPPROTO_IPV6, IPV6_V6ONLY, &opt, sizeof(opt)) < 0) { perror("setsockopt"); close(sockfd); return -1; } // 加入多播组,以接收设备发现响应 struct ip_mreq mreq; memset(&mreq, 0, sizeof(mreq)); mreq.imr_multiaddr.s_addr = inet_addr(multicast_addr); mreq.imr_interface.s_addr = htonl(INADDR_ANY); if (setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0) { perror("setsockopt"); close(sockfd); return -1; } // 加入ipv6多播组,以接收设备发现响应 struct ipv6_mreq mreq_v6; memset(&mreq_v6, 0, sizeof(mreq_v6)); if (inet_pton(AF_INET6, multicast_addr_v6, &mreq_v6.ipv6mr_multiaddr) < 0) { perror("inet_pton"); close(sockfd); return -1; } mreq_v6.ipv6mr_interface = 0; if (setsockopt(sockfd, IPPROTO_IPV6, IPV6_JOIN_GROUP, &mreq_v6, sizeof(mreq_v6)) < 0) { perror("setsockopt"); close(sockfd); return -1; } // 设置socket为非阻塞模式 int flags = fcntl(sockfd, F_GETFL, 0); if (flags < 0) { perror("fcntl"); close(sockfd); return -1; } flags |= O_NONBLOCK; if (fcntl(sockfd, F_SETFL, flags) < 0) { perror("fcntl"); close(sockfd); return -1; } // 发送设备发现请求 struct sockaddr_in6 dest_addr; memset(&dest_addr, 0, sizeof(dest_addr)); dest_addr.sin6_family = AF_INET6; dest_addr.sin6_port = htons(multicast_port); if(inet_pton(AF_INET6, multicast_addr_v6, &dest_addr.sin6_addr) < 0) { perror("inet_pton"); close(sockfd); return -1; } if (sendto(sockfd, "M-SEARCH * HTTP/1.1\r\n\r\n", 25, 0, (struct sockaddr*)&dest_addr, sizeof(dest_addr)) < 0) { perror("sendto"); close(sockfd); return -1; } // 接收设备发现响应 const int buf_size = 1024; char buf[buf_size]; struct sockaddr_storage src_addr; socklen_t src_addr_len; src_addr_len = sizeof(src_addr); while (1) { int n = recvfrom(sockfd, buf, buf_size - 1, 0, (struct sockaddr*)&src_addr, &src_addr_len); if (n < 0) { if (errno == EAGAIN || errno == EWOULDBLOCK) { // 没有数据可接收,继续等待 continue; } else { perror("recvfrom"); close(sockfd); return -1; } } buf[n] = '\0'; // 解析设备发现响应中的IP地址和端口号 char ip_str[INET6_ADDRSTRLEN]; int port; if (src_addr.ss_family == AF_INET) { struct sockaddr_in *sin = (struct sockaddr_in *)&src_addr; inet_ntop(AF_INET, &(sin->sin_addr), ip_str, INET_ADDRSTRLEN); port = ntohs(sin->sin_port); } else if (src_addr.ss_family == AF_INET6) { struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)&src_addr; inet_ntop(AF_INET6, &(sin6->sin6_addr), ip_str, INET6_ADDRSTRLEN); port = ntohs(sin6->sin6_port); } else { continue; } // 输出设备发现响应中的IP地址和端口号 cout << "Device found: " << ip_str << ":" << port << endl; } // 关闭socket close(sockfd); soap_destroy(&soap); soap_end(&soap); soap_done(&soap); return 0; } ``` 这个示例代码使用了IPv6socket来同时支持IPv4IPv6。使用setsockopt()函数分别设置IPV6_MULTICAST_IF和IPV6_V6ONLY选项,然后分别加入IPv4IPv6多播组,以接收设备发现响应。在接收响应时,需要根据地址族(IPv4IPv6)来解析IP地址和端口号。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值