1 套接字地址结构
struct sockaddr{ unsigned short sa_family;//地址类型 char sa_data[14]; //14字节的地址协议}
sa_family表示套接字的协议族类型,对应于TCP/IP的协议该是AF_INET;
与该结构体等价的另一个数据结构:sockaddr_in。
struct sockaddr_in
{
unsigned short sin_family; //地址类型
unsigned short sin_port ; //端口号
struct in_addr sin_addr ;//IP地址
unsigned char sin_zero[8];//填充字节,一般为0
}
struct in_addr {
unsigned long s_addr;
};
结构体sockaddr和结构体sockaddr_in都是16个字节。通常在编写基于TCP/IP协议的网络程序的时候,使用sockaddr_in来设置地址,然后通过类型的强制转换成sockaddr类型。
struct sockaddr_in sock;
sock.sin_family = AF_INET;
sock.sin_port = htons(80);
sock.sin_addr.s_addr = inet_addr("202.112.12.11");
memset(sock.sin_zero, 0, sizeof(sock.sin_zero)); //将数组sin_zero设置成为0
2 创建套接字
#include <sys/types.h>
#include <sys/socket.h>
int socket(int domain, int type, int protocol);
domain用来表示套接字所使用的协议族。AF_UNIX:在本机通讯的套接字;AF_INET:使用ipv4; AF_INET6:使用ipv6。
type指示套接字的类型。SOCK_STREAM SOCK_DGRAM SOCK_RAW
3 建立连接
函数connect用来在一个执行的套接字上穿件一个连接。
#include <sys/types.h>
#include <sys/socket.h>
int connect(int sockfd, const struct sockaddr *serv_addr, socklen_t addrlen);
服务器的IP地址和端口号由serv_addr来制定。
4 绑定套接字
int bind(int sockfd, struct sockaddr *my_addr, socklen_t addrlen);
这个函数只有服务器端使用。
5 在套接字上侦听
int listen(int s, int backlog);
在服务器端,一般先创建一个socket,然后调用函数bind将该套接字绑定到某个端口上,接着调用listen函数等待来自于客户端的连接请求。
一般多个客户端连接到一个服务器,服务器设置一个连接队列,记录已经建立的连接,参数backlog指定了该连接队列的最大长度。如果连接队列已经达到最大,
之后的连接请求将会被服务器拒绝。
6 接受连接
int accept(int s, struct sockaddr *addr, socklen_t *addrlen);
参数s是由函数socket创建,经函数bind绑定到本地某一端口上,然后通过函数listen转化而来的监听套接字。
参数addr是用来保存发起连接请求的主机的地址和端口。
参数addrlen是addr所指向的结构体的大小。
只能对面向连接的套接字使用accept函数。accept成功时,将创建一个新的套接字,并为这个新的套接字分配一个新的套接字描述符,类似文件描述符,进程可以利用这个新的套接字描述符与客户端交换数据。参数S指定的套接字将继续等待客户端的连接请求。
如果参数s指向的套接字被设置成为阻塞方式(Linux默认的设置),且连接请求队列为空的时候,则accept函数将被阻塞直到有连接请求为止;
如果参数s指向的套接字被设置成为非阻塞方式(fcntl),如果队列为空,则accept立即返回-1,errno设置成为EAGAIN。
7 TCP套接字 发送数据
size_t send(int s, const void *msg, size_t len, int flags);
函数send只能对处于连接状态的套杰斯使用。参数S为已建立好连接的套接字描述符,即accept函数的返回值。
参数msg指向存放待发数据的缓冲区,参数len为待发数据的长度。
如果要发送的数据太长而不能发送时,将出现错误,error设置成为EMSGSIZE;
如果要发送的数据长度大于该套接字的缓冲区剩余空间大小时,send()一般会被阻塞,如果该套接字设置成为非阻塞方式,则此时立即返回-1.
执行成功则返回实际发送的数据的字节数。
执行成功只能说明数据写入套接字的缓冲区中,并不表示数据已经成功的通过网络发送到目的地。
8 接受数据
size_t recv(int s, coid *buf, size_t len, int flags);
如果一个数据包太长以至于缓冲区放不下的时候,剩余部分的数据将可能被丢弃。
如果在指定的套接字上无数据到达时,recv函数将被阻塞,如果套接字设置成为非阻塞的话,则立即返回-1;