本文共 4723 字,大约阅读时间需要 15 分钟。
Linux下:
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下相同
#includeint listen(SOCKET s,int backlog);->成功时返回0,失败时返回 INVALID_SOCKET
4. 调用connect函数从客户端发送连接请求 与Linux下相同
#includeint connect(SOCKET s,const struct sockaddr* name,int namelen);->成功时返回0,失败时返回SOCKET_ERROR
5. 关闭套接字(对标Linux下的close()函数)
#includeint closesocket(SOCKET s);->成功时返回0,失败时返回SOCKET_ERROR
以上五个就是基于Windows下的套接字相关函数,虽然返回值和参数与Linux函数不同,但是具有相同功能的函数名称是一样的。这就很方便呀~
Linux内部也将套接字当作文件,因此,不管创建文件还是套接字都返回文件描述符。
之前也通过示例介绍了文件描述符返回及编号的过程。 Windows中通过调用系统函数创建文件时,返回“句柄”( handle),换言之, Windows中的句柄相当于 Linux中的文件描述符。只不过 Windows中要区分文件句柄和套接字句柄。
虽然都称为“句柄”,但不像 Linux那样完全一致。文件句柄相关函数与套接字句柄相关函数是有区别的,这一点不同于 Linux文件描述符。既然对句柄有了一定理解,接下来再观察基于 Windows的套接字相关函数,这将加深各位对
SOCKET
类型的参数和返回值的理解。 的确!它就是为了保存套接字句柄整型值的新数据类型,
它由 typedef 声明定义。回顾 是socket、 listen和 accept等套接字相关函数,则更能体会到与 Linux中套接字相关函数的相似性。接下来将之前基于 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);}
注意!:别忘了传入参数哦~ 不然可连不上~
Linux中套接字也是文件,因此可以通过文件I/O函数read 和 write进行数据传输。 windows中这有些不同。
windows严格区分文件I/O函数和套接字I/O函数 下面是win的数据传输函数1. send()函数
#includeint 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()函数
#includeint recv(SOCKET s,const char* buf,int len,int flags);->成功时返回接收的字节数,(收到 EOF 则返回0),失败返回 SOCKET_ERROR
转载地址:http://hjzwi.baihongyu.com/