目录
一、socket()函数 (创建套接字)
#include <sys/types.h> /* See NOTES */
#include <sys/ socket.h>
int socket (int domain, int type, int protocol);
domain 是地址族
PF_INET // internet 协议
PF_UNIX // unix internal协议
PF_NS // Xerox NS协议
PF_IMPLINK // Interface Message协议
type // 套接字类型
SOCK_STREAM // 流式套接字 唯一对应TCP
SOCK_DGRAM // 数据报套接字 唯一对应UDP
SOCK_RAW // 原始套接字
protocol 参数通常置为0 原始套接字编程时需填充
返回值:
RETURN VALUE
On success,a file descriptor for the new socket is returned. On error,-1 is returned,and errno is set appropriately.
成功时返回文件描述符,出错时返回为-1
二、bind()函数
SYNOPSIS
#include <sys/types.h> /*see NOTES */
#include <sys/ socket.h>
int bind (int sockfd, struct sockaddr* addr, int addrLen);
struct sockaddr_in
u_short sin_family // protocol family
u_short sin_port // port number
struct in_addr sin_addr //IP address (32-bits)
返回值: On success, zero is returned. On error, -1 is returned, and errno is
set appropriately.
示例代码:
/*2. 绑定 */
/*2.1 填充struct sockaddr_in结构体变量 */
bzero (&sin, sizeof (sin));
sin.sin_family = AF_INET;
sin.sin_port = htons (SERV_PORT); //网络字节序的端口号
/*优化1: 让服务器程序能绑定在任意的IP上 */
#if 1
sin.sin_addr.s_addr = htonl (INADDY_ANY);
#else
if (inet_pton (AF_INET, SERV_IP_ADDR, (void *) &sin.sin_addr) != 1) {
perror ("inet_pton");
exit (1);
}
#endif
/*2.2 绑定 */
if (bind (fd, (struct sockaddr *) &sin, sizeof (sin)) < 0) {
perror ("bind");
exit (1);
}
如果是IPVG的编 程,要使用struct sockdr_in6结构体(详细情况请参考man 7 ipv6),通常更通用的方法可以通过struct sockaddr_torage来编程
地址结构的一般用法:
定义一个struct sockaddr_in类型的变量并清空
struct sockaddr_in myaddr;
memset(&myaddr, 0, sizeof(myaddr));
填充地址信息
myaddr.sin_family = PF_INET;
myaddr.sin_port = htons(8888);
myaddr.sin_addr.s_addr = inet_addr(“192.168.1.100”);
将该变量强制转换为struct sockaddr类型在函数中使用
bind(listenfd, (struct sockaddr*)(&myaddr), sizeof(myaddr));
地址转换函数:
unsigned long inet_addr(char *address);
address是以’\0’结尾的点分IPv4字符串。该函数返回32位的地址。如果字符串包含的不是合法的IP地址,则函数返回-1。例如:
struct in_addr addr;
addr.s_addr = inet_addr(" 192.168.1.100 ");
char* inet_ntoa(struct in_addr address);
address是IPv4地址结构,函数返回一指向包含点分IP地址的静态存储区字符指针。如果错误则函数返回NULL
三、listen()函数 把主动套接字变成被动套接字
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
int listen (int sockfd, int backlog);
sockfd:监听连接的套接字(通过socket()函数拿到的fd)
backlog: 一般填5 。同时允许几路客户端和服务器进行正在连接的过程(正在三次握手)一般填5,测试得知,ARM最大为8
内核中服务器的套接字fd会维护2个链表;
1.正在三次握手的的客户端链表(数量=2*backlog+1)
2.已经建立好连接的客户端链表(已经完成3次握手分配好了newfd)
比如: listen(fd, 5); //表示系统允许11 (=2*5+1)个客户端同时进行三次握手
指定了正在等待连接的最大队列长度,它的作用在于处理可能同时出现的几个连接请求。
DoS(拒绝服务)攻击即利用了这个原理,非法的连接占用了全部的连接数,造成正常的连接请求被拒绝。
返回值: 成功0 或 失败-1
四、accept()函数 阻塞等待客户端连接请求
•int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen) ;
#include <sys/types.h>
#include <sys/socket.h>
listen()和accept()是TCP服务器端使用的函数
五、connect() 客户端的连接函数
SYNOPSIS
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);
返回值:0 或 -1
头文件:
#include <sys/types.h>
#include <sys/socket.h>
sockfd : socket返回的文件描述符
serv_addr : 服务器端的地址信息
addrlen : serv_addr的长度
connect()函数和服务器的bind()函数写法类似;
connect()是客户端使用的系统调用。
六、编程实例
net.h:
#ifndef __MAKEU_NET_H__
#define __MAKEU_NET_H__
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <strings.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include<arpa/inet.h>
#define SERV_PORT 5001
#define SERV_IP_ADDR "127.0.0.1"
#define BACKLOG 5
#define QUIT_STR "quit"
#endif
server.c:
#include "net.h"
int main (void)
{
int fd = -1;
struct sockaddr_in sin;
/* 1. 创建socket fd */
if ((fd = socket (AF_INET, SOCK_STREAM, 0)) < 0) {