Socket编程实践(2) --Socket编程导引

什么是Socket?

    Socket可以看成是用户进程内核网络协议栈的接口(编程接口, 如下图所示), 其不仅可以用于本机进程间通信,可以用于网络上不同主机的进程间通信, 甚至还可以用于异构系统之间的通信。

 

IPv4套接口地址结构

    IPv4套接口地址结构通常也称为“网际套接字地址结构”,它以“sockaddr_in”命名,定义在头文件<netinet/in.h>中

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //TCP/IP地址结构  
  2. struct sockaddr_in  
  3. {  
  4.     uint8_t  sin_len;  
  5.     sa_family_t  sin_family;  
  6.     in_port_t   sin_port;   //2字节  
  7.     struct in_addr  sin_addr;   //4字节  
  8.     char sin_zero[8];   //8字节  
  9. };  

成员说明:

    sin_len:整个sockaddr_in结构体的长度,在4.3BSD-Reno版本之前的第一个成员是sin_family.

    sin_family:指定该地址家族,对于IPv4来说必须设为AF_INET

    sin_port:端口

    sin_addr:IPv4的地址;

    sin_zero:暂不使用,一般将其设置为0

 

Linux结构(常用):

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. struct sockaddr_in  
  2. {  
  3.     sa_family_t    sin_family; /* address family: AF_INET */  
  4.     in_port_t      sin_port;   /* port in network byte order(网络字节序) */  
  5.     struct in_addr sin_addr;   /* internet address */  
  6. };  
  7. /* Internet address. */  
  8. struct in_addr  
  9. {  
  10.     uint32_t       s_addr;     /* address in network byte order */  
  11. };  

通用地址结构

用来指定与套接字关联的地址(可以支持其他协议).

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. struct sockaddr  
  2. {  
  3.     uint8_t  sin_len;  
  4.     sa_family_t  sin_family;  
  5.     char sa_data[14];   //14字节     
  6. };  

说明:

    sin_len:整个sockaddr结构体的长度

    sin_family:指定该地址家族

    sa_data:由sin_family决定它的形式。

 

网络字节序

1.大端字节序(Big Endian)

    最高有效位(MSB:Most Significant Bit)存储于最低内存地址处,最低有效位(LSB:Lowest Significant Bit)存储于最高内存地址处。

2.小端字节序(Little Endian)

    最高有效位(MSB:Most Significant Bit)存储于最高内存地址处,最低有效位(LSB:Lowest Significant Bit)存储于最低内存地址处。

3.主机字节序

    不同的主机有不同的字节序,如x86为小端字节序,Motorola 6800为大端字节序,ARM字节序是可配置的。

4.网络字节序

    网络字节序规定为大端字节序

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //测试当前系统是否为小端模式  
  2. int main()  
  3. {  
  4.     int data = 0x12345678;  //int = 4字节(32位)  
  5.                             //每4个二进制位代表1位十六进制位,  
  6.                             //则8位十六进制位代表4*8=32位二进制位  
  7.     char *p = (char *)&data;  
  8.     printf("%x, %x, %x, %x\n",p[0],p[1],p[2],p[3]);  
  9.   
  10.     //0x78属于低位,如果其放在了p[0](低地址)处,则说明是小端模式  
  11.     if (p[0] == 0x78)  
  12.     {  
  13.         cout << "当前系统为小端模式" << endl;    //x86平台为小端模式  
  14.     }  
  15.     else if (p[0] == 0x12)  
  16.     {  
  17.         cout << "当前系统为大端模式" << endl;    //IBM为大端模式  
  18.     }  
  19. }  

字节序转换函数(常用于端口转换)

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. uint32_t htonl(uint32_t hostlong);  
  2. uint16_t htons(uint16_t hostshort);  
  3. uint32_t ntohl(uint32_t netlong);  
  4. uint16_t ntohs(uint16_t netshort);  
  5. /**说明: 
  6. h代表(local)host;n代表network; 
  7. s代表short;l代表long; 
  8. */  
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //测试转换结果  
  2. int main()  
  3. {  
  4.     int localeData = 0x12345678;  
  5.     char *p = (char *)&localeData;  
  6.     printf("Begin: %0x %0x %0x %0x\n", p[0], p[1], p[2], p[3]);  
  7.     //将本地字节转换成网络字节  
  8.     int inetData = htonl(localeData);  
  9.     p = (char *)&inetData;  
  10.     printf("After: %0x %0x %0x %0x\n", p[0], p[1], p[2], p[3]);  
  11.   
  12.     if (p[0] == 0x12)  
  13.         cout << "网络系统为大端模式" << endl;  
  14.     else  
  15.         cout << "网络系统为小端模式" << endl;  
  16.     printf("host:%x, inet:%x\n", localeData, inetData);  
  17. }  

地址转换函数(用于IP地址转换)

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <netinet/in.h>  
  2. #include <arpa/inet.h>  
  3. int inet_aton(const char *cp, struct in_addr *inp);  
  4. in_addr_t inet_addr(const char *cp);      
  5. char *inet_ntoa(struct in_addr in);  
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //in_addr定义如下:  
  2. typedef uint32_t in_addr_t;  
  3. struct in_addr  
  4. {  
  5.     in_addr_t s_addr;  
  6. };  


[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. //实践  
  2. int main()  
  3. {  
  4.     //将点分十进制转换成十进制数  
  5.     cout << inet_addr("192.168.139.137") << endl;  
  6.   
  7.     //将十进制数转换成点分十进制形式  
  8.     struct in_addr address;  
  9.     address.s_addr = inet_addr("192.168.139.137");  
  10.     cout << inet_ntoa(address) << endl;  
  11.   
  12.     memset(&address,0,sizeof(address));  
  13.     inet_aton("127.0.0.1", &address);  
  14.     cout << address.s_addr << endl;  
  15.     cout << inet_ntoa(address) << endl;  
  16.     return 0;  
  17. }  

套接字类型

1)流式套接字(SOCK_STREAM)

    提供面向连接的、可靠的数据传输服务,数据无差错,无重复的发送,且按发送顺序接收, 对应TCP协议。

2)数据报式套接字(SOCK_DGRAM)

    提供无连接服务。不提供无错保证,数据可能丢失或重复,并且接收顺序混乱, 对应UDP协议。

3)原始套接字(SOCK_RAW)

    使我们可以跨越传输层直接对IP层进行封装传输.

 

TCP客户/服务器模型 


简单echo服务器模型

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值