转载自: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. centos编译安装rabbitmq

    安装环境 [root@VM_12_50_centos rabbitmq]# uname -a Linux VM_12_50_centos 3.10.0-514.21.1.el7.x86_64 #1 S ...

  2. 003---socket介绍

    socket介绍 什么是socket? socket是应用层与tcp/ip协议族通信的中间软件抽象层,它是一组接口.在设计模式中.其实就是一个门面模式.我们无需深入理解tcp/udp协议,socket ...

  3. 基于Ubuntu Server 16.04 LTS版本安装和部署Django之(四):安装MySQL数据库

    基于Ubuntu Server 16.04 LTS版本安装和部署Django之(一):安装Python3-pip和Django 基于Ubuntu Server 16.04 LTS版本安装和部署Djan ...

  4. Python:正则表达式—— re 模块

    一.什么是正则表达式(Regular Expression) 正则表达式本身是一种小型的.高度专业化的编程语言,它内嵌在Python中,并通过 re(regular expression)模块实现.使 ...

  5. python的正则表达一

    一.常用的正则表达式 1.了解正则表达式 正则表达式是对字符串操作的一种逻辑公式,就是用事先定义好的一些特定字符.及这些特定字符的组合,组成一个“规则字符串”,这个“规则字符串”用来表达对字符串的一种 ...

  6. hdu4742 Pinball Game 3D

    真他娘的搞不懂cdq分治的顺序问题.但是candy?的博客里提到过,多想想吧-- #include <algorithm> #include <iostream> #inclu ...

  7. 纯js生成QRCode

    纯js,不依赖jquery,非常好用,废话不多说,直接上代码! <!DOCTYPE html> <html> <head> <meta charset=&qu ...

  8. Cassandra 数据库设计

    Cassandra 2.* CQL3.1 最近更新:2015-10-30 索引的设计 在Cassandra中经常会发现,索引不够用,不好用,各种不强大. 比如,我关注的人的需求uid + follow ...

  9. Struts2—整合Spring

    Struts2—整合Spring Spring框架是一个非常优秀的轻量级java EE容器,大部分javaEE应用,都会考虑使用Spring容器来管理应用中的组件. Struts2是一个MVC框架,是 ...

  10. POJ 1703 Find them, Catch them(并查集拓展)

    Description The police office in Tadu City decides to say ends to the chaos, as launch actions to ro ...