windows下的C++ socket服务器(3)
int make_server_socket(int port)
{
WSADATA inet_WsaData;//1
WSAStartup(MAKEWORD(2, 0), &inet_WsaData);//1
1 WSADATA inet_WsaData;SAStartup(MAKEWORD(1, 1), &inet_WsaData);
在windows下使用socket的相关函数前,必须通过WSAStartup函数完成对Winsock服务的初始化。
int WSAStartup(WORD wVersionRequested,LPWSADATA lpWSAData);
该函数的第一个参数指明程序请求使用的Socket版本,其中高位字节指明副版本、低位字节指明主版本;第二个参数可以用来返回请求的Socket的版本信息。当一个应用程序调用WSAStartup函数时,操作系统根据请求的Socket版本来搜索相应的Socket库,然后绑定找到的Socket库到该应用程序中。以后应用程序就可以调用所请求的Socket库中的其它Socket函数了。
以前大家使用的都是socket1.1版本,但socket2.0版本已经出来了,所以我这里使用的是socket2.0版本(MAKEWORD(2.0))
1.1版和2.0版的区别:
两者的最重要区别是1.1版只支持TCP/IP协议,而2.0版可以支持多协议。2.0版有良好的向后兼容性,任何使用1.1版的源代码,二进制文件,应用程序都可以不加修改地在2.0规范下使用。
MAKEWORD的定义如下
#define MAKEWORD(a, b) ((WORD)(((BYTE)(((DWORD_PTR)(a)) & 0xff)) | ((WORD)((BYTE)(((DWORD_PTR)(b)) & 0xff))) << 8))
2 if (LOBYTE(inet_WsaData.wVersion) != 2 || HIBYTE(inet_WsaData.wVersion) != 0)
用于检测当前的Socket是否为2.0
LOBYTE和HIBYTE是两个宏,在vs2013里定义如下
#define LOBYTE(w) ((BYTE)(((DWORD_PTR)(w)) & 0xff))
#define HIBYTE(w) ((BYTE)((((DWORD_PTR)(w)) >> 8) & 0xff))
WSACleanup();用于解除与Socket库的绑定并释放Socket库所占用的系统资源。
3 int tcp_socket = socket(AF_INET, SOCK_STREAM, 0);
socket函数用于建立一个socket,函数原型如下
SOCKET socket(int af, int type, int protocol);
第一个参数af指定应用程序使用的通信协议的协议族,af一般置为AF_INET(表示internetwork: UDP, TCP等);
第二个参数type为协议的Socket类型,常用的有3种:SOCK_STREAM、SOCK_DGRAM和SOCK_RAW。
SOCK_STREAM对应于TCP。
SOCK_DGRAM对应于UDP。
SOCK_RAW称为原始Socket,可以读写ICMP、IGMP、IP报文。前两种类型使用得最多。
第三个参数protocol指定所使用的协议。对于SOCK_STREAM、SOCK_DGRAM两种类型的Socket,该参
数为0,对于原始Socket才需要指定具体的协议。
4 struct sockaddr_in saddr;
sockaddr_in是定义了socket发送和接收数据包的地址的结构体,有四个字段,含义如下
第一个参数short sin_family,指定应用程序使用的通信协议的协议族,af一般置为AF_INET(表示internetwork: UDP, TCP, etc.Internetwork Version 4);
第二个参数u_short sin_port,代表程序使用的IP地址端口,由程序员指定;
第三个参数struct in_addr sin_addr中的s_addr,用于设置IP地址;
第四个参数char sin_zero[8],是为了保证sockaddr_in与SOCKADDR类型的长度相等而填充进来的字段。
例如
struct sockaddr_in saddr;
saddr.sin_family = AF_INET;
saddr.sin_port = htons(port);//使用的端口号
saddr.sin_addr.s_addr = INADDR_ANY;//任意地址均可以,这样任意客户端都可以访问到服务器
5 ::bind(tcp_socket, (const struct sockaddr*)&saddr, sizeof(saddr))
这里使用::表示的位于全局作用域下的bind,由于我之前使用了using namespace std;所以如果没有使用::,它会使用std下的bind,出现一系列的错误
bind函数用来将一个socket套接字绑定到一个地址,很多函数会隐式的调用bind函数。
bind的函数原型如下
int bind(SOCKET s,const struct sockaddr FAR * name,int namelen);
第一个参数指定待绑定的Socket描述符;
第二个参数指绑定到的地址结构,即一个sockaddr类型的数据;
第三个参数指对应的是地址的大小;
如果bind错误,返回-1,
例如
if (::bind(tcp_socket, (const struct sockaddr*)&saddr, sizeof(saddr)) == -1)//绑定到tcp_socket,使用saddr的地址结构,该地址的大小为sizeof(saddr),
{
cerr << "bind error" << endl;
return -1;
}
6 ::listen(tcp_socket, 1)
如果作为一个服务器,在调用socket()、bind()之后需要调用listen()来监听这个socket,如果客户端这时调用connect()发出连接请求,服务器端就会接收到这个请求。
listen的函数原型如下
int listen(SOCKET s,int backlog);
第一个参数为要监听的socket描述字;
第二个参数为相应socket可以排队的最大连接个数。()(当客户链接请求大于这数时(即缓冲池满),其它的未进入链接缓冲池的客户端在tcp层上tcp模块会自动重新链接,直到超时(大约57秒后))
如果listen错误,返回-1,
- Android OpenGL开发实践 - GLSurfaceView对摄像头数据的再处理
- 走进科学:对七夕“超级病毒”XX神器的逆向分析
- 机器学习 - 朴素贝叶斯分类器的意见和文本挖掘
- 认知指纹:颠覆性的身份认证技术
- 跟我学姿势:极客教你如何科学的看电影
- Discuz 5.x/6.x/7.x投票SQL注入分析
- 论如何高效的挖掘漏洞
- Rxjava + retrofit + dagger2 + mvp搭建Android框架
- 走进科学:如何正确的隐藏自己的行踪
- 比特儿(Bter.com) 比特币交易平台被盗事件全解析
- BitTorrent Bleep:无法被监控的聊天软件
- QQ蠕虫的行为检测方法
- 趋势OfficeScan系列产品漏洞分析
- [置顶] 浅谈我为什么选择用Retrofit作为我的网络请求框架
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法