申明:这是一个仿muduo库的项目,在这里只截取一些重要的部分进行仿写~
InetAddress是用于封装socket地址类型
#pragma once
#include <arpa/inet.h>
#include <netinet/in.h>
#include <string>
//封装socket地址类型
class InetAddress
{
public:
explicit InetAddress(uint16_t port, std::string ip = "127.0.0.1");
explicit InetAddress(const sockaddr_in &addr)
:addr_(addr)
{}
std::string toIp() const;
std::string toIpPort() const;
uint16_t toPort() const;
const sockaddr_in* getSockAddr() const { return &addr_;}
private:
sockaddr_in addr_;
};
其内部实现如下所示
#include "InetAddress.h"
#include <strings.h>
#include <string.h>
InetAddress::InetAddress(uint16_t port, std::string ip)
{
bzero(&addr_, sizeof addr_);
addr_.sin_family=AF_INET; //选择IPv4网络协议
addr_.sin_port = htons(port); //本地字节序转网络字节序【大小端问题->转为大端字节序】
addr_.sin_addr.s_addr = inet_addr(ip.c_str()); //将十进制IP转为长整型
}
std::string InetAddress::toIp() const
{
//addr_
char buf[64] = {0};
::inet_ntop(AF_INET,&addr_.sin_addr,buf,sizeof buf); //将网络传输的二进制数值转化为成点分十进制的ip地址
return buf;
}
std::string InetAddress::toIpPort() const
{
//ip:port
char buf[64] = {0};
::inet_ntop(AF_INET,&addr_.sin_addr,buf,sizeof buf);
size_t end = strlen(buf);
uint16_t port = ntohs(addr_.sin_port); //网络字节序转本地字节序
sprintf(buf+end,":%u",port);
return buf;
}
uint16_t InetAddress::toPort() const
{
return ntohs(addr_.sin_port);
}
/*
#include <iostream>
int main()
{
InetAddress addr(8080);
std::cout<<addr.toIpPort()<<std::endl;
return 0;
}
*/
ntohs =net to host short int 16位
htons=host to net short int 16位
ntohl=net to host long int 32位
htonl=host to net long int 32位网络字节顺序NBO(Network Byte Order)
按从高到低的顺序存储,【大端序】,在网络上使用同一的网络字节顺序,可避免兼容性问题;
主机字节顺序HBO(Host Byte Order)
不同的机器HBO不相同,与CPU设计有关,数据顺序是由CPU决定的,而与操作系统无关;
由于这个原因,不同体系结构的机器之间不能直接通信,所以要转换成一种约定的顺序,也就是网络字节顺序。在PC开发中有ntohl和htonl函数可以用来进行网络字节和主机字节的转换htons的功能:将一个无符号短整型数值转换为网络字节序,即大端模式(big-endian)
1、把ip地址转化为用于网络传输的二进制数值
int inet_aton(const char *cp, struct in_addr *inp);
inet_aton() 转换网络主机地址ip(如192.168.1.10)为二进制数值,并存储在struct in_addr结构中,即第二个参数*inp,函数返回非0表示cp主机有地有效,返回0表示主机地址无效。(这个转换完后不能用于网络传输,还需要调用htons或htonl函数才能将主机字节顺序转化为网络字节顺序)
in_addr_t inet_addr(const char *cp);
inet_addr函数转换网络主机地址(如192.168.1.10)为网络字节序二进制值,如果参数char *cp无效,函数返回-1(INADDR_NONE),这个函数在处理地址为255.255.255.255时也返回-1,255.255.255.255是一个有效的地址,不过inet_addr无法处理;
2、将网络传输的二进制数值转化为成点分十进制的ip地址
char *inet_ntoa(struct in_addr in);
inet_ntoa 函数转换网络字节排序的地址为标准的ASCII以点分开的地址,该函数返回指向点分开的字符串地址(如192.168.1.10)的指针,该字符串的空间为静态分配的,这意味着在第二次调用该函数时,上一次调用将会被重写(复盖),所以如果需要保存该串最后复制出来自己管理!
3.新型网路地址转化函数inet_pton和inet_ntop
这两个函数是随IPv6出现的函数,对于IPv4地址和IPv6地址都适用,
函数中p和n分别代表表达(presentation)和数值(numeric)。地址的表达格式通常是ASCII字符串,数值格式则是存放到套接字地址结构的二进制值。