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

服务端代码:

  1. #include <iostream>
  2. #include <string>
  3. #include <sys/types.h>
  4. #include <sys/socket.h>
  5. #include <arpa/inet.h>
  6. #include <netdb.h>
  7. #include <errno.h>
  8. #include <time.h>
  9. using namespace std;
  10. int tcp_listen(const char *host, const char *service, const int listen_num = 5)
  11. {
  12. int listenfd, ret;
  13. const int on = 1;
  14. struct addrinfo hints, *res, *ressave;
  15. bzero(&hints, sizeof(hints));
  16. hints.ai_flags = AI_PASSIVE;
  17. hints.ai_family = AF_UNSPEC;
  18. hints.ai_socktype = SOCK_STREAM;
  19. hints.ai_protocol = IPPROTO_IP;
  20. if (0 != (ret = getaddrinfo(host, service, &hints, &res)))
  21. {
  22. cout << "getaddrinfo error: " << gai_strerror(ret) << endl;
  23. return -1;
  24. }
  25. ressave = res;
  26. while(NULL != res)
  27. {
  28. if (-1 == (listenfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)))
  29. {
  30. cout << "create socket error: " << strerror(errno) << endl;
  31. res = res->ai_next;
  32. continue;
  33. }
  34. if (-1 == setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)))
  35. {
  36. cout << "setsockopt error: " << strerror(errno) << endl;
  37. close(listenfd);
  38. res = res->ai_next;
  39. continue;
  40. }
  41. if (-1 == bind(listenfd, res->ai_addr, res->ai_addrlen))
  42. {
  43. cout << "bind error: " << strerror(errno) << endl;
  44. close(listenfd);
  45. res = res->ai_next;
  46. continue;
  47. }
  48. if (-1 == listen(listenfd, listen_num))
  49. {
  50. cout << "listen error: " << strerror(errno) << endl;
  51. close(listenfd);
  52. res = res->ai_next;
  53. continue;
  54. }
  55. break;
  56. }
  57. freeaddrinfo(ressave);
  58. if (NULL == res)
  59. return -1;
  60. return listenfd;
  61. }
  62. int get_addrinfo(const struct sockaddr *addr, string &ip, in_port_t &port)
  63. {
  64. void *numeric_addr = NULL;
  65. char addr_buff[INET6_ADDRSTRLEN];
  66. if (AF_INET == addr->sa_family)
  67. {
  68. numeric_addr = &((struct sockaddr_in*)addr)->sin_addr;
  69. port = ntohs(((struct sockaddr_in*)addr)->sin_port);
  70. }
  71. else if (AF_INET6 == addr->sa_family)
  72. {
  73. numeric_addr = &((struct sockaddr_in6*)addr)->sin6_addr;
  74. port = ntohs(((struct sockaddr_in6*)addr)->sin6_port);
  75. }
  76. else
  77. {
  78. return -1;
  79. }
  80. if (NULL != inet_ntop(addr->sa_family, numeric_addr, addr_buff, sizeof(addr_buff)))
  81. ip = addr_buff;
  82. else
  83. return -1;
  84. return 0;
  85. }
  86. int main(int argc, char *argv[])
  87. {
  88. int listenfd, connfd;
  89. struct sockaddr_storage cliaddr;
  90. socklen_t len = sizeof(cliaddr);
  91. time_t now;
  92. char buff[128];
  93. if (2 == argc) //指定端口
  94. listenfd = tcp_listen(NULL, argv[1]);
  95. else if (3 == argc) //指定本地IP和端口
  96. listenfd = tcp_listen(argv[1], argv[2]);
  97. else
  98. {
  99. cout << "usage: " << argv[0] << " [<hostname/ipaddress>] <service/port>" << endl;
  100. return -1;
  101. }
  102. if (listenfd < 0)
  103. {
  104. cout << "call tcp_listen error" << endl;
  105. return -1;
  106. }
  107. while (true)
  108. {
  109. connfd = accept(listenfd, (struct sockaddr*)&cliaddr, &len);
  110. string ip = "";
  111. in_port_t port = 0;
  112. get_addrinfo((struct sockaddr*)&cliaddr, ip, port);
  113. cout << "client " << ip << "|" << port << " login" << endl;
  114. now = time(NULL);
  115. snprintf(buff, sizeof(buff) - 1, "%.24s", ctime(&now));
  116. write(connfd, buff, strlen(buff));
  117. close(connfd);
  118. }
  119. close(listenfd);
  120. return 0;
  121. }

客户端代码:

  1. #include <iostream>
  2. #include <string>
  3. #include <sys/types.h>
  4. #include <sys/socket.h>
  5. #include <arpa/inet.h>
  6. #include <netdb.h>
  7. #include <errno.h>
  8. #include <time.h>
  9. using namespace std;
  10. int tcp_connect(const char *host, const char *service)
  11. {
  12. int sockfd, ret;
  13. struct addrinfo hints, *res, *ressave;
  14. bzero(&hints, sizeof(hints));
  15. hints.ai_family = AF_UNSPEC;
  16. hints.ai_socktype = SOCK_STREAM;
  17. hints.ai_protocol = IPPROTO_IP;
  18. if (0 != (ret = getaddrinfo(host, service, &hints, &res)))
  19. {
  20. cout << "getaddrinfo error: " << gai_strerror(ret) << endl;
  21. return -1;
  22. }
  23. ressave = res;
  24. while (NULL != res)
  25. {
  26. if (-1 == (sockfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)))
  27. {
  28. cout << "create socket error: " << strerror(errno) << endl;
  29. res = res->ai_next;
  30. continue;
  31. }
  32. if (-1 == connect(sockfd, res->ai_addr, res->ai_addrlen))
  33. {
  34. cout << "connect error: " << strerror(errno) << endl;
  35. close(sockfd);
  36. res = res->ai_next;
  37. continue;
  38. }
  39. break;
  40. }
  41. freeaddrinfo(ressave);
  42. if (NULL == res)
  43. return -1;
  44. return sockfd;
  45. }
  46. int main(int argc, char *argv[])
  47. {
  48. int sockfd, n;
  49. char buff[128];
  50. struct sockaddr_storage cliaddr;
  51. if (3 != argc)
  52. {
  53. cout << "usage: " << argv[0] << " <hostname/ipaddress> <service/port>" << endl;
  54. return -1;
  55. }
  56. sockfd = tcp_connect(argv[1], argv[2]);
  57. if (sockfd < 0)
  58. {
  59. cout << "call tcp_connect error" << endl;
  60. return -1;
  61. }
  62. bzero(buff, sizeof(buff));
  63. while ((n = read(sockfd, buff, sizeof(buff) - 1) > 0))
  64. {
  65. cout << buff << endl;
  66. bzero(buff, sizeof(buff));
  67. }
  68. close(sockfd);
  69. return 0;
  70. }

编译:

g++ daytimesrv.cpp -o daytimesrv

g++ daytimecli.cpp -o daytimecli

运行举例:

服务端运行情况:

客户端运行情况:

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

 
 

C++ IPv4与IPv6的兼容编码(转,出自http://blog.csdn.net/ligt0610/article/details/18667595)的更多相关文章

  1. Sublime Text3 离线安装Package Control并使用GBK编码 --转自https://blog.csdn.net/swhard/article/details/78930371

    1.关闭Sublime Text 3,去https://github.com/wbond/package_control/releases下载一个zip包,我下载的是 2.将包内的顶层文件夹解压至C: ...

  2. 稀疏编码(Sparse Coding)的前世今生(一) 转自http://blog.csdn.net/marvin521/article/details/8980853

    稀疏编码来源于神经科学,计算机科学和机器学习领域一般一开始就从稀疏编码算法讲起,上来就是找基向量(超完备基),但是我觉得其源头也比较有意思,知道根基的情况下,拓展其应用也比较有底气.哲学.神经科学.计 ...

  3. JavaScript在IE和Firefox(火狐)的不兼容问题解决方法小结 【转】http://blog.csdn.net/uniqer/article/details/7789104

    1.兼容firefox的 outerHTML,FF中没有outerHtml的方法. 代码如下: if (window.HTMLElement) { HTMLElement.prototype.__de ...

  4. ubuntu16.04 tomcat7安装和编码修改(转发:https://blog.csdn.net/zl544434558/article/details/76735564)

    有直接通过命令安装的,但是我还是喜欢把文件下载下来,然后自己配置. 1,下载tomcat7二进制文件 https://tomcat.apache.org/download-70.cgi 2,解压tom ...

  5. IPv4和IPv6的兼容问题

    一网络拓扑 Ipv6网络1 路由器A IPv4网络 路由器B IPv6网络2 二知识补充 [注]双协议栈主机(路由器A.B)通过域名解析器区分传过来的是IPv4还是IPv6 三处理技术 双协议栈 Ip ...

  6. ios 兼容IPV4和IPV6网络通信

    前言: 苹果官方出了新的规定,要求新上架的app都必须单独支持ipv6-only的网络. 准备工作: 搭建IPV6测试环境:http://blog.csdn.net/potato512/article ...

  7. VLC测试IPv4 IGMP/IPv6 MLD协议

    1 简述 VLC既可以充当流客户端,又可以充当流服务器,并且可以跨平台使用,是一款开源.免费的软件,基于GNU GPL许可证. 2 搭建组播服务器 第一步:运行程序后选择“媒体  串流”:第二步:通过 ...

  8. [转载]Linux网络编程IPv4和IPv6的inet_addr、inet_aton、inet_pton等函数小结

    转载:http://blog.csdn.net/ithomer/article/details/6100734 知识背景: 210.25.132.181属于IP地址的ASCII表示法,也就是字符串形式 ...

  9. Linux网络编程IPv4和IPv6的inet_addr、inet_aton、inet_pton等函数小结(转)

    原文:http://blog.csdn.net/ithomer/article/details/6100734 知识背景: 210.25.132.181属于IP地址的ASCII表示法,也就是字符串形式 ...

随机推荐

  1. G - Harmonic Number (II) 找规律--> 给定一个数n,求n除以1~n这n个数的和。n达到2^31 - 1;

    /** 题目:G - Harmonic Number (II) 链接:https://vjudge.net/contest/154246#problem/G 题意:给定一个数n,求n除以1~n这n个数 ...

  2. Vsphere日记02.ESXi5.5.configuration

    2.Vsphere ESXi 5.5 configuration 步骤1:启动 ESXI 服务器 步骤2:按 F2 进入用户登录界面 输入用户名及密码,进入参数设置界面.密码为我安装时候设置的&quo ...

  3. 关于在Java中链接SQLServer数据库中失败的原因分析

    首先声明:笔者是Java的初学者,并且一值是走在自学的道路上,长久以来只有“度娘”相伴.(加入了各种Java学习群,基本没有热心帮人解决问题的.可以理解-_-!!!)大神级的人物就不必看拙文了,没有什 ...

  4. python 设计模式之单例模式

    单例模式就是防止每个请求到来,都需要在内存里创建一个实例,再通过该实例执行指定的方法. 如果并发量大的话,内存里就会存在非常多功能上一模一样的对象.存在这些对象肯定会消耗内存,对于这些功能相同的对象可 ...

  5. Java NIO(2):缓冲区基础

    缓冲区(Buffer)对象是面向块的I/O的基础,也是NIO的核心对象之一.在NIO中每一次I/O操作都离不开Buffer,每一次的读和写都是针对Buffer操作的.Buffer在实现上本质是一个数组 ...

  6. linux系统启动过程具体解释-开机加电后发生了什么 --linux内核剖析(零)

    本文參考了例如以下文章 深入理解linux启动过程 mbr (主引导记录(Master Boot Record)) 电脑从开机加电到操作系统main函数之前执行的过程 详细解释linux系统的启动过程 ...

  7. sedna进行xquery查询

    有一个文件book.xml: <books> <book> <name>The Call Of Wild</name> <author>Ja ...

  8. TCP和UDP 协议发送数据包的大小

    在进行UDP编程的时候,我们最容易想到的问题就是,一次发送多少bytes好? 当然,这个没有唯一答案,相对于不同的系统,不同的要求,其得到的答案是不一样的,这里仅对像ICQ一类的发送聊天消息的情况作分 ...

  9. PHP中mysql_fetch_row()、mysql_fetch_assoc()和mysql_fetch_array()的联系

    总是记不住或者混淆mysql_fetch_row().mysql_fetch_assoc()和mysql_fetch_array()这三个函数的朋友们注意了,今天我在这里给大家总结一下他们之间的关系, ...

  10. android菜鸟学习笔记30----Android使用百度地图API(一)准备工作及在应用中显示地图

    1.准备工作: 百度地图API是免费开放的,但是需要申请API Key: 1)先注册一个百度开发者帐号 2)进入百度开放服务平台http://developer.baidu.com/ 3)进入LBS云 ...