博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Part 1 start network programming:chapter one:1.4 基于Windows的套接字相关函数及示例
阅读量:3936 次
发布时间:2019-05-23

本文共 4723 字,大约阅读时间需要 15 分钟。

1.4 基于Windows的套接字相关函数及示例

1.4.1 Win与Linux下的对标

Linux下:

  1. 调用socket函数创建套接字
  2. 调用bind函数分配ip地址和端口号
  3. 调用listen函数转为可接受请求状态
  4. 调用accept函数受理连接请求

Windows下函数介绍:

1. 创建套接字的函数 socket() 与Linux下不同

SOCKET socket(int af, int type , int protocol);->成功时返回套接字句柄 (linux下叫做描述符,是int类型),失败时返回INVALID_SOCKET

2. 为套接字分配ip地址和端口号的 bind()函数 与Linux下相同

int bind(SOCKET s, const struct sockaddr* name,int namelen);->成功时返回0,失败时返回 SOCKET_ERROR

3.使套接字可以接收客户端连接(也就是说让这个端成为了主机端) 与Linux下相同

#include 
int listen(SOCKET s,int backlog);->成功时返回0,失败时返回 INVALID_SOCKET

4. 调用connect函数从客户端发送连接请求 与Linux下相同

#include 
int connect(SOCKET s,const struct sockaddr* name,int namelen);->成功时返回0,失败时返回SOCKET_ERROR

5. 关闭套接字(对标Linux下的close()函数)

#include 
int closesocket(SOCKET s);->成功时返回0,失败时返回SOCKET_ERROR

以上五个就是基于Windows下的套接字相关函数,虽然返回值和参数与Linux函数不同,但是具有相同功能的函数名称是一样的。这就很方便呀~

1.4.2 Win中的文件句柄和套接字句柄

Linux内部也将套接字当作文件,因此,不管创建文件还是套接字都返回文件描述符

之前也通过示例介绍了文件描述符返回及编号的过程。
Windows中通过调用系统函数创建文件时,返回“句柄”( handle),换言之, Windows中的句柄相当于 Linux中的文件描述符。

只不过 Windows中要区分文件句柄和套接字句柄

虽然都称为“句柄”,但不像 Linux那样完全一致。文件句柄相关函数与套接字句柄相关函数是有区别的,这一点不同于 Linux文件描述符。

既然对句柄有了一定理解,接下来再观察基于 Windows的套接字相关函数,这将加深各位对

SOCKET类型的参数和返回值的理解。

的确!它就是为了保存套接字句柄整型值的新数据类型,

它由 typedef 声明定义。回顾 是socket、 listen和 accept等套接字相关函数,则更能体会到与 Linux中套接字相关函数的相似性。

1.4.3 创建基于 Windows的服务器端和客户端

接下来将之前基于 Linux的服务器端与客户端示例转化到 Windows平台。完全理解有些困难,我们只需验证套接字相关函数的调用过程、套接字库的初始化与注销过程即可。

下面按照服务器端和客户端分别介绍

1. 服务器端 hello_server_win.c

#include 
#include
#include
void ErrorHandling(char* message);int main(int argc, char* argv[]){
WSADATA wsaData; SOCKET hServSock, hClntSock; SOCKADDR_IN servAddr,clntAddr; int szClntAddr; char message[] = "Hello world!"; if(argc != 2){
printf("Usage %s
\n", argv[0]); exit(1); } if(WSAStartup(MAKEWORD(2, 2), &wsaData) != 0){
//初始化Winsock库 ErrorHandling("WSAStartup() error !"); } hServSock = socket(PF_INET, SOCK_STREAM, 0); //创建套接字 if(hServSock == INVALID_SOCKET){
ErrorHandling("socket() error!"); } memset(&servAddr, 0, sizeof(servAddr)); servAddr.sin_family = AF_INET; servAddr.sin_addr.s_addr = htonl(INADDR_ANY); servAddr.sin_port = htons(atoi(argv[1])); //为此套接字分配ip地址和端口号 if(bind(hServSock, (SOCKADDR*) &servAddr, sizeof(servAddr)) ==SOCKET_ERROR){
ErrorHandling("bind() error"); } if(listen(hServSock, 5) == SOCKET_ERROR){
//通过listen函数使之成为服务器端套接字 ErrorHandling ("listen() error"); } szClntAddr = sizeof(clntAddr); //接受来自客户端的连接请求 hClntSock = accept(hServSock, (SOCKADDR*)&clntAddr, &szClntAddr); if(hClntSock == INVALID_SOCKET){
ErrorHandling("accept() error"); } send(hClntSock, message, sizeof(message), 0); //send函数,向客户端传输数据 closesocket(hClntSock); closesocket(hServSock); WSACleanup(); //程序终止前,注销初始化的套接字库 return 0;}void ErrorHandling(char* message){
fputs(message,stderr); fputc('\n',stderr); exit(1);}

2. 客户端代码 hello_client_win.c

#include 
#include
#include
void ErrorHandling(char* message);int main(int argc, char* argv[]){
WSADATA wsaData; SOCKET hSocket; SOCKADDR_IN servAddr; char message[30]; int strlen; if(argc != 3){
printf("Usage %s
\n", argv[0]); exit(1); } if(WSAStartup(MAKEWORD(2, 2), &wsaData) != 0){
//初始化Winsock库 ErrorHandling("WSAStartup() error !"); } hSocket = socket(PF_INET, SOCK_STREAM, 0); //创建套接字 if(hSocket == INVALID_SOCKET){
ErrorHandling("socket() error!"); } memset(&servAddr, 0, sizeof(servAddr)); servAddr.sin_family = AF_INET; servAddr.sin_addr.s_addr = inet_addr(argv[1]); servAddr.sin_port = htons(atoi(argv[2])); //通过此套接字向服务器端发出连接请求 if(connect(hSocket, (SOCKADDR*)&servAddr,sizeof(servAddr)) == SOCKET_ERROR){
ErrorHandling ("connect() error"); } //用recv函数接受服务器发来的数据 strlen = recv(hSocket,message,sizeof(message) -1,0); if(strlen == -1){
ErrorHandling("read() error!"); } printf("Message from server: %s \n",message ); closesocket(hSocket); WSACleanup(); //程序终止前,注销初始化的套接字库 return 0;}void ErrorHandling(char* message){
fputs(message,stderr); fputc('\n',stderr); exit(1);}

注意!:别忘了传入参数哦~ 不然可连不上~

1.4.4 基于windows的I/O函数

Linux中套接字也是文件,因此可以通过文件I/O函数read 和 write进行数据传输。 windows中这有些不同。

windows严格区分文件I/O函数和套接字I/O函数 下面是win的数据传输函数

1. send()函数

#include 
int send(SOCKET s, const char* buf,int len,int flags

);

->成功时返回0,失败时返回SOCKET_ERROR

参数说明:

s:表示数据传输对象连接的套接字句柄值(给谁传)
buf:保存待传输数据的缓冲地址(传什么)
len:要传输的字节(传多少)
flags:传输数据时用到的多种选项信息

这个函数与Linux中的write函数 只多了一个flags。后面会对这个函数详细说明,现在flags = 0就行

send并非win独有,linux下也有,只不过这里是为了强调一下在li下socket可以视为文件

2. recv()函数

#include 
int recv(SOCKET s,const char* buf,int len,int flags);->成功时返回接收的字节数,(收到 EOF 则返回0),失败返回 SOCKET_ERROR

转载地址:http://hjzwi.baihongyu.com/

你可能感兴趣的文章
七层协议图
查看>>
华为交换机作为AC的条件
查看>>
禁用Ubuntu 15.04登录界面显示客人会话(简单-实用)
查看>>
linux X下安装的软件
查看>>
Linux监测某一时刻对外的IP连接情况
查看>>
CentOS7 最小环境安装Jumpserver 1.0版脚本
查看>>
X-Security X的安全控制
查看>>
openVAS的安装
查看>>
Centos 6.5 初始安装无网卡驱动解决方法
查看>>
linux中的网桥bridge
查看>>
linux中的teaming与bonding
查看>>
LVM
查看>>
用shell切分文件--split
查看>>
python中判断字符是否为中文
查看>>
Python - 利用zip函数将两个列表(list)组成字典(dict)
查看>>
python-全角转半角
查看>>
Python pass语句作用与用法
查看>>
Java double,float设置小数点位数
查看>>
PyCharm & Jupyter
查看>>
为什么要用Jupyter Notebook
查看>>