在讲解套接字编程函数之前,有必要对socket编程的两个不可或缺的结构体进行说明。
第一个结构体式struct sockaddr.。这个结构为许多类型的套接字储存套接字地址信息:
#include<sys/socket.h>
struct sockaddr {
uint8_t sa_len;
unsigned short sa_family; /* 地址家族, AF_xxx */
char sa_data[14]; /*14字节协议地址*/
};
由于历史的原因,套接字函数中(如connect,bind等)使用的参数类型大多是sockaddr类型的。而如今进行套接字编程的时候大都使用sockaddr_in进行套接字地址填充
struct sockaddr_in {
uint8_t sa_len; /* 结构体长度*/
short int sin_family; /* 通信类型 */
unsigned short int sin_port; /* 端口 */
struct in_addr sin_addr; /* Internet 地址 */
unsigned char sin_zero[8]; /* 未使用的*/
};
struct in_addr { //sin_addr的结构体类型in_addr 原型
unsigned long s_addr; /*存4字节的 IP 地址(使用网络字节顺序)。*/
};
因此,这就要求对这些函数进行调用的时候都必须要讲套接字地址结构指针进行类型强制转换,例如:
struct sockaddr_in serv;
bind(sockfd,(struct sockaddr *)&serv,sizeof(serv));
否则C编译器会产生警告信息(把不兼容的指针类型传递给“bind”函数的第二个参数)。(注意sockaddr_in的sin_port和 sin_addr 必须是网络字节顺序 (Network Byte Order))
connect() 系统调用函数原型如下所示:
#include <sys/types.h>;
#include <sys/socket.h>;
int connect(int sockfd, struct sockaddr *serv_addr, int addrlen);
sockfd 是系统调用 socket() 返回的套接字文件描述符。serv_addr 是 保存着目的地端口和 IP 地址的数据结构 struct sockaddr。addrlen 设置 为 sizeof(struct sockaddr)。
connect函数在调用失败的时候返回值-1,并会设置全局错误变量 errno。
如果是TCP套接字,调用connect会激发TCP的三路握手过程,其出错返回设置errno变量值有如下几种:
(1) 若TCP没有收到SYN分节的响应,则返回 会设置 errno变量值为ETIMEDOUT。
(2) 若对客户的SYN的响应是RST(表示复位),则表明该服务器主机在我们制定的端口上没有进程在等待与之连接(例如服务器进程或许没有在运行),此时errno 会设置为ECONNREFUSED错误。
(3) 若客户发出的SYN在中间的某个路由器引发了一个ICMP错误报文(例如主机不可达)。客户主机内核会保存该信息,并按一定的时间间隔继续发送SYN,如若在某个规定得时间内还没收到响应,则会把保存的消息作为EHOSTUNREACH返回给进程。