转载自: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. hive表格取差集

    hive 求两个集合的差集 业务场景是这样的,这里由两个hive表格A和B A的形式大概是这样的:uid B的形式大概是这样的:uid 我想要得到存在A中但是不存在B中的uid 具体代码如下 sele ...

  2. mysql5.6主主复制及keepalived 高可用

    1.实验目的 mysql服务器作为生产环境中使用最广泛的数据库软件,以其开源性,稳定性而广泛使用,但同时由于数据存储,读写频率高,极易造成数据库出错,从而给企业造成不可挽回的损失,我们除了做好数据库的 ...

  3. R语言绘图:ROC曲线图

    使用pROC包绘制ROC曲线 #####***绘制ROC曲线***##### library("pROC") N <- dim(data2)[1] #数据长度 set.see ...

  4. HDU暑假多校第八场J-Taotao Picks Apples

    一.题意 给定一个序列,之后给出若干个修改,修改的内容为在原序列的基础上,将某一位元素的值改成给定的值<每次修改相互独立,不保存修改后的结果>.之后询问,在选择第一位元素的情况下,最长递增 ...

  5. C#的特性Attribute

    一.什么是特性 特性是用于在运行时传递程序中各种元素(比如类.方法.结构.枚举.组件等)的行为信息的声明性标签,这个标签可以有多个.您可以通过使用特性向程序添加声明性信息.一个声明性标签是通过放置在它 ...

  6. delphi 数据库中Connection与Query连接数量问题思考

    今天闲着没事,测试了一下Connection连接MSSQL,可以承受多少连接.    1.看看ADOConnection的连接数:写了一个代码,动态创建,测试了10000个连接,花了大约5~10分钟创 ...

  7. LeetCode:12. Integer to Roman(Medium)

    1. 原题链接 https://leetcode.com/problems/integer-to-roman/description/ 2. 题目要求 (1) 将整数转换成罗马数字: (2) 整数的范 ...

  8. 为什么在默认情况下无法修改被block捕获的变量? __block都做了什么?

    默认情况下,block里面的变量,拷贝进去的是变量的值,而不是指向变量的内存的指针.使用__block修饰后的变量,拷贝到block里面的就是指向变量的指针,所以我们就可以修改变量的值.

  9. 一次和别人争吵一个按钮,点击后显示导航;再点击不显示的效果,是否一定以及必须用js?

    事情经过是这样的,我们组一个说话很喜欢用一定,肯定的哥们,吃午饭的时候拿了自己做的一个UI库,头部有一个按钮 点击展开,再次点击收缩,他意思说一个按钮无法记录点击状态,必须使用js.然后我看了一眼,心 ...

  10. Java并发基础--多线程基础

    一.多线程基础知识 1.进程和线程 进程:是指一个内存中运行的应用程序,每个进程都有一个独立的内存空间,一个应用程序可以同时运行多个进程:进程也是程序的一次执行过程,是系统运行程序的基本单位:系统运行 ...