转载自:http://blog.csdn.net/neicole/article/details/7459021

一。程序运行效果图

二。程序源代码

三。程序设计相关基础知识

1.计算机网络
    2.IP地址
    3.协议
    4.网络体系结构
    5.TCP/IP体系结构与特点
    6.客户机/服务器模式
    7.TCP/IP特点
    8.套接字的引入
    9.面向 连接/无连接 的套接字的系统调用时序图/流程图

一。程序运行效果图

二。程序源代码

  1. // server.cpp
  2. #include <iostream>
  3. #include <cstdio>
  4. #include <Winsock2.h>
  5. using namespace std;
  6. int main()
  7. {
  8. // 加载socket动态链接库(dll)
  9. WORD wVersionRequested;
  10. WSADATA wsaData;    // 这结构是用于接收Wjndows Socket的结构信息的
  11. int err;
  12. wVersionRequested = MAKEWORD( 1, 1 );   // 请求1.1版本的WinSock库
  13. err = WSAStartup( wVersionRequested, &wsaData );
  14. if ( err != 0 ) {
  15. return -1;          // 返回值为零的时候是表示成功申请WSAStartup
  16. }
  17. if ( LOBYTE( wsaData.wVersion ) != 1 || HIBYTE( wsaData.wVersion ) != 1 ) {
  18. // 检查这个低字节是不是1,高字节是不是1以确定是否我们所请求的1.1版本
  19. // 否则的话,调用WSACleanup()清除信息,结束函数
  20. WSACleanup( );
  21. return -1;
  22. }
  23. // 创建socket操作,建立流式套接字,返回套接字号sockSrv
  24. // SOCKET socket(int af, int type, int protocol);
  25. // 第一个参数,指定地址簇(TCP/IP只能是AF_INET,也可写成PF_INET)
  26. // 第二个,选择套接字的类型(流式套接字),第三个,特定地址家族相关协议(0为自动)
  27. SOCKET sockSrv = socket(AF_INET, SOCK_STREAM, 0);
  28. // 套接字sockSrv与本地地址相连
  29. // int bind(SOCKET s, const struct sockaddr* name, int namelen);
  30. // 第一个参数,指定需要绑定的套接字;
  31. // 第二个参数,指定该套接字的本地地址信息,该地址结构会随所用的网络协议的不同而不同
  32. // 第三个参数,指定该网络协议地址的长度
  33. // PS: struct sockaddr{ u_short sa_family; char sa_data[14];};
  34. //                      sa_family指定该地址家族, sa_data起到占位占用一块内存分配区的作用
  35. //     在TCP/IP中,可使用sockaddr_in结构替换sockaddr,以方便填写地址信息
  36. //
  37. //     struct sockaddr_in{ short sin_family; unsigned short sin_port; struct in_addr sin_addr; char sin_zero[8];};
  38. //     sin_family表示地址族,对于IP地址,sin_family成员将一直是AF_INET。
  39. //     sin_port指定将要分配给套接字的端口。
  40. //     sin_addr给出套接字的主机IP地址。
  41. //     sin_zero[8]给出填充数,让sockaddr_in与sockaddr结构的长度一样。
  42. //     将IP地址指定为INADDR_ANY,允许套接字向任何分配给本地机器的IP地址发送或接收数据。
  43. //     如果想只让套接字使用多个IP中的一个地址,可指定实际地址,用inet_addr()函数。
  44. SOCKADDR_IN addrSrv;
  45. addrSrv.sin_addr.S_un.S_addr = htonl(INADDR_ANY); // 将INADDR_ANY转换为网络字节序,调用 htonl(long型)或htons(整型)
  46. addrSrv.sin_family = AF_INET;
  47. addrSrv.sin_port = htons(6000);
  48. bind(sockSrv, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR)); // 第二参数要强制类型转换
  49. // 将套接字设置为监听模式(连接请求), listen()通知TCP服务器准备好接收连接
  50. // int listen(SOCKET s,  int backlog);
  51. // 第一个参数指定需要设置的套接字,第二个参数为(等待连接队列的最大长度)
  52. listen(sockSrv, 10);
  53. // accept(),接收连接,等待客户端连接
  54. // SOCKET accept(  SOCKET s,  struct sockaddr* addr,  int* addrlen);
  55. // 第一个参数,接收一个处于监听状态下的套接字
  56. // 第二个参数,sockaddr用于保存客户端地址的信息
  57. // 第三个参数,用于指定这个地址的长度
  58. // 返回的是向与这个监听状态下的套接字通信的套接字
  59. // 客户端与用户端进行通信
  60. // send(), 在套接字上发送数据
  61. // int send( SOCKET s,  const char* buf,  int len,  int flags);
  62. // 第一个参数,需要发送信息的套接字,
  63. // 第二个参数,包含了需要被传送的数据,
  64. // 第三个参数是buffer的数据长度,
  65. // 第四个参数,一些传送参数的设置
  66. // recv(), 在套接字上接收数据
  67. // int recv(  SOCKET s,  char* buf,  int len,  int flags);
  68. // 第一个参数,建立连接后的套接字,
  69. // 第二个参数,接收数据
  70. // 第三个参数,接收数据的长度,
  71. // 第四个参数,一些传送参数的设置
  72. SOCKADDR_IN  addrClient;
  73. int len = sizeof(SOCKADDR);
  74. while(true){    // 不断等待客户端请求的到来
  75. SOCKET sockConn = accept(sockSrv, (SOCKADDR*)&addrClient, &len);
  76. char sendBuf[100];
  77. sprintf(sendBuf, "Welcome %s to the server program~ \nNow, let's start talking...\n", inet_ntoa(addrClient.sin_addr));
  78. send(sockConn, sendBuf, strlen(sendBuf)+1, 0);  // 发送显示欢迎信息
  79. char recvBuf[100];
  80. recv(sockConn, recvBuf, 100, 0);
  81. printf("%s\n", recvBuf);        // 接收第一次信息
  82. char * sockConnName = "Client";
  83. printf("我们可以聊五句话");
  84. int n = 5;
  85. while(n--){
  86. printf("还剩%d次:\n", n+1);
  87. char recvBuf[100];
  88. recv(sockConn, recvBuf, 100, 0);
  89. printf("%s Says: %s\n", sockConnName, recvBuf);     // 接收信息
  90. char talk[100];
  91. printf("Please enter what you want to say next(\"quit\"to exit):");
  92. gets(talk);
  93. send(sockConn, talk, strlen(talk)+1, 0);            // 发送信息
  94. printf("\n");
  95. }
  96. printf("\nEnd talking... \n");
  97. closesocket(sockConn);
  98. }
  99. printf("\n");
  100. system("pause");
  101. return 0;
  102. }
  1. // client.cpp
  2. #include <iostream>
  3. #include <cstdio>
  4. #include <Winsock2.h>
  5. using namespace std;
  6. int main()
  7. {
  8. // 加载socket动态链接库(dll)
  9. WORD wVersionRequested;
  10. WSADATA wsaData;    // 这结构是用于接收Wjndows Socket的结构信息的
  11. int err;
  12. wVersionRequested = MAKEWORD( 1, 1 );   // 请求1.1版本的WinSock库
  13. err = WSAStartup( wVersionRequested, &wsaData );
  14. if ( err != 0 ) {
  15. return -1;          // 返回值为零的时候是表示成功申请WSAStartup
  16. }
  17. if ( LOBYTE( wsaData.wVersion ) != 1 || HIBYTE( wsaData.wVersion ) != 1 ) {
  18. // 检查这个低字节是不是1,高字节是不是1以确定是否我们所请求的1.1版本
  19. // 否则的话,调用WSACleanup()清除信息,结束函数
  20. WSACleanup( );
  21. return -1;
  22. }
  23. // 创建socket操作,建立流式套接字,返回套接字号sockClient
  24. // SOCKET socket(int af, int type, int protocol);
  25. // 第一个参数,指定地址簇(TCP/IP只能是AF_INET,也可写成PF_INET)
  26. // 第二个,选择套接字的类型(流式套接字),第三个,特定地址家族相关协议(0为自动)
  27. SOCKET sockClient = socket(AF_INET, SOCK_STREAM, 0);
  28. // 将套接字sockClient与远程主机相连
  29. // int connect( SOCKET s,  const struct sockaddr* name,  int namelen);
  30. // 第一个参数:需要进行连接操作的套接字
  31. // 第二个参数:设定所需要连接的地址信息
  32. // 第三个参数:地址的长度
  33. SOCKADDR_IN addrSrv;
  34. addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");      // 本地回路地址是127.0.0.1;
  35. addrSrv.sin_family = AF_INET;
  36. addrSrv.sin_port = htons(6000);
  37. connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR));
  38. char recvBuf[100];
  39. recv(sockClient, recvBuf, 100, 0);
  40. printf("%s\n", recvBuf);
  41. send(sockClient, "Attention: A Client has enter...\n", strlen("Attention: A Client has enter...\n")+1, 0);
  42. printf("我们可以聊五句话");
  43. int n = 5;
  44. do{
  45. printf("\n还剩%d次:", n);
  46. char talk[100];
  47. printf("\nPlease enter what you want to say next(\"quit\"to exit):");
  48. gets(talk);
  49. send(sockClient, talk, strlen(talk)+1, 0);          // 发送信息
  50. char recvBuf[100];
  51. recv(sockClient, recvBuf, 100, 0);
  52. printf("%s Says: %s\n", "Server", recvBuf);     // 接收信息
  53. }while(--n);
  54. printf("End linking...\n");
  55. closesocket(sockClient);
  56. WSACleanup();   // 终止对套接字库的使用
  57. printf("\n");
  58. system("pause");
  59. return 0;
  60. }

三。程序设计相关基础知识

1.计算机网络
2.IP地址
3.协议
4.网络体系结构

5.TCP/IP体系结构与特点

6.客户机/服务器模式

7.TCP/IP特点

8.套接字的引入

9.面向 连接/无连接 的套接字的系统调用时序图/流程图

如果出现编译错误,需要在编译选项里加入“-lwsock32”

Windows Socket 编程_ 简单的服务器/客户端程序的更多相关文章

  1. Windows Socket 编程_单个服务器对多个客户端简单通讯

    单个服务器对多个客户端程序: 一.简要说明 二.查看效果 三.编写思路 四.程序源代码 五.存在问题 一.简要说明: 程序名为:TcpSocketOneServerToMulClient 程序功能:实 ...

  2. 5.1 socket编程、简单并发服务器

    什么是socket? socket可以看成是用户进程与内核网络协议栈的编程接口.是一套api函数. socket不仅可以用于本机的进程间通信,还可以用于网络上不同主机间的进程间通信. 工业上使用的为t ...

  3. UNIX网络编程——TCP回射服务器/客户端程序

    下面通过最简单的客户端/服务器程序的实例来学习socket API. serv.c 程序的功能是从客户端读取字符然后直接回射回去: #include<stdio.h> #include&l ...

  4. c++下基于windows socket的单线程服务器客户端程序(基于TCP协议)

    今天自己编写了一个简单的c++服务器客户端程序,注释较详细,在此做个笔记. windows下socket编程的主要流程可概括如下:初始化ws2_32.dll动态库-->创建套接字-->绑定 ...

  5. 运用socket实现简单的服务器客户端交互

    Socket解释: 网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket. Socket的英文原义是“孔”或“插座”.作为BSD UNIX的进程通信机制,取后一种意 ...

  6. c++下基于windows socket的服务器客户端程序(基于UDP协议)

    前天写了一个基于tcp协议的服务器客户端程序,今天写了一个基于UDP协议的,由于在上一篇使用TCP协议的服务器中注释已经较为详细,且许多api的调用是相同的,故不再另外注释. 使用UDP协议需要注意几 ...

  7. Windows socket之最简单的socket程序

    原文:Windows socket之最简单的socket程序 最简单的服务器的socket程序流程如下(面向连接的TCP连接 ): 1. WSAStartup(); 初始化网络库的使用. 2. soc ...

  8. python socket 实现的简单http服务器

    预备知识: 关于http 协议的基础请参考这里. 关于socket 基础函数请参考这里. 关于python 网络编程基础请参考这里. 一.python socket 实现的简单http服务器   废话 ...

  9. socket编程,简单多线程服务端测试程序

    socket编程,简单多线程服务端测试程序 前些天重温了MSDN关于socket编程的WSAStartup.WSACleanup.socket.closesocket.bind.listen.acce ...

随机推荐

  1. Python全栈day 02

    Python全栈day 02 一.循环语句 while 用法 num = 1 while num <= 10: print(num) num += 1 # 循环打印输出1-10 while el ...

  2. Python学习笔记:第一天python基础

    目录 1. python简介 2. python的安装 3. 编写第一个helloword 4. 变量和常量 5. 数据类型 6. 输入 7. if语句 1. python简介 python是在198 ...

  3. Ruby字符串的一些方法

    最近因为公司需求开始看ruby,先从ruby的基本数据类型开始看 看到ruby的字符串类型string,发现ruby中的字符串单双引号是不一样的,这点和Python有那么点不一样 主要是我们对字符串进 ...

  4. mtools使用-1

    mtools是什么? mtools 是一组非常好用的 MongoDB 日志分析工具 ,由MongoDB Inc 官方工程师所写. 组成部分 mlogfilter :按时间切片日志文件,合并日志文件,过 ...

  5. HyperLedger Fabric 1.4 超级账本起源(5.1)

    至比特币开源以来,无数技术人员对其进行研究,并且对该系统经过了无数次改进,超级账本项目(Hyperledger)最初也是用来改善比特币的底层技术,最终由Linux基金会组织发展起来.       开放 ...

  6. Fiddler 发送post 请求失败

    今天服务端同事,让我发一个post 请求.然后呢,一直有问题.告诉我签名失败. 后来换了其他的在线模拟post,都是可以的. 后来找到原因了, post 请求,必须要有Content-Type 和 C ...

  7. samba与apache配置使用

    samba与apache根目录 1.直接将apache用户作为samba用户 2.给apache用户赋宇网站根目录的acl rwx权限 #注意 第一次不要加默认的权限 setfacl -m u:apa ...

  8. FJOI 2019 游记

    (FJOI 2019 滚粗记) Day 0 早上刷了一些水题,然后就上路了. (画图3D好好玩) 来得晚只有十几分钟时间看考场,于是连试机题都没有Ak. Day 1 果然我还是太菜了 走过来的时候再讨 ...

  9. Linux-获得命令帮助man

    date:显示当前系统时间,修改时间 clock,hwclock:显示硬件时间 cal:calendar,查看日历 计时器靠晶体振荡器来完成计时 Linux: 实时时钟,rtc,real time c ...

  10. 9.0 toast定位+WebDriverWait显示等待

    Toast  判断-----基本操作问题 首先基本操作,进入安卓市场的账号密码页面--- from appium import webdriver from selenium.webdriver.su ...