<网络编程>套接字介绍

1、端口:IANA(Internet Assigned Numbers Authority)维护着一个端口号分配状况的清单。

  • 众所周知的端口(0-1023):由IANA分配和控制,可能的话,相同的端口号尽可能分配给TCP,UDP和STCP的同一给定服务
  • 已登记的端口(1024-49151):这些端口不受IANA控制,
  • 动态、私用的端口(49152-65535):临时端口。

2、套接字:一个套接字对是一个定义该连接的两个端点的四元组。{本地IP,本地TCP端口号,外地IP,外地TCP端口号}

套接字对唯一标识一个网络上的每个TCP连接。

3、套接字地址结构

大多数套接字函数需要一个指定套接字地址结构的指针作为参数,每个协议族都定义了自己的套接字结构。套接字地址结构仅在给定主机上使用,虽然结构中的某些字段用在不同主机之间的通信,但是结构本身并不在主机之间传递。

unp-3-1

  • sockaddr_in
    • IPv4地址和TCP或者UDP端口号在套接字地址结构中以网络字节序存储。
    • sin_falily:可以是无符号整数类型,在支持长度字段的实现中,通常是8位无符号整数,不支持长度字段中是16位无符号整数
    • sin_addr:in_addr_t必须至少是32位的无符号整数类型
    • sin_port:in_port_t必须至少是16位的无符号整数类型
    • 长度是16
  • sockaddr_in6:长度是28
  • sockaddr_storage相比于sockaddr优势:
    • 足够大,能够容纳系统支持的任何套接字结构
    • 如果系统支持的任何套接字地址结构有对齐的需要,那么sockaddr_storage能够满足最苛刻的对齐要求

4、值-结果参数

  • 从进程到内核传递套接字地址结构:bind,connect,sendto
  • 从内核到进程传递套接字地址结构:accept,recvfrom,getsockname,getpeername
  • 值:告诉内核该结构的大小,内核在写结构的时候,不至于越界。
  • 结果:告诉进程内核在该结构中实际存储了多少信息。(如果套接字地址结构是固定长度的,那么从内核返回的值总是那个固定长度,如IPv4的sockaddr_in长度是16,IPv6的sockaddr_in6长度是28;对于可变长度的套接字地址结构,返回值可能小于结构的最大长度)

5、字节函数

  • 字节序
    • 小端字节序:高序字节存储在高地址,低序字节存储在低地址。
    • 大端字节序:高序字节存储在低地址,低序字节存储在高地址
  • 主机字节序:某个给定系统所用的字节序。
  • 网络字节序:网络协议必须指定的一个网络字节序,网络协议使用大端字节序。

主机字节序和网络字节序之间的转换函数:

#include <netinet/in.h>
//主机-->网络字节序
uint16_t htons(uint16_t host16bitvalue);
uint32_t htons(uint32_t host32bitvalue);
//网络字节序-->主机
uint16_t ntohs(uint16_t net16bitvalue);
uint32_t ntohs(uint32_t net32bitvalue);
其中h代表host,n代表network,s代表short,l代表long

字节操纵函数:

/*第一组:起源于4.2BSD,几乎所有现今支持套接字函数的系统仍然提供*/
#include <strings.h>
void bzero(void *dest,size_t nbytes);
void bcopy(const void *src,void *dest,size_t nbytes);
int bcmp(const void *ptr1,const void *ptr2,size_t nbytes);

/***********************************************************************
 *  第二组:起源于ANSI C,支持ANSI C函数库的所有系统都提供
 **********************************************************************/
#include <string.h>
void* memset(void *dest,int c,size_t len);
void* memcpy(void *dest,const void *src,size_t nbytes);
int memcmp(const void *ptr1,const void *ptr2,size_t nbytes);

地址转换函数:

#include <arpa/inet.h>
//第一组
/*
    strptr:指向c字符串,是一个点分十进制的地址
    addrptr/inaddr:网络字节序二进制值
    inet_addr函数:如今已废弃,新代码应该使用inet_aton(该函数出错时,
                       返回INADDR_NONE,32位均为1,因此255.255.255.255
                     不能由该函数处理)
    inet_ntoa函数:参数传入的是结构而不是结构的指针;

*/
int inet_aton(const char *strptr, struct in_addr *addrptr);
int_addr_t inet_addr(const char *strptr);
char* inet_ntoa(struct in_addr inaddr);

//第二组
int inet_pton(int family,const char *strptr,void *addrptr);//成功返回1,字符串无效返回0,出错-1
const char* inet_ntop(int family,const void *addrptr,char *strptr,size_t len);

猜你喜欢

转载自www.cnblogs.com/tingweichen/p/10705289.html