高性能分布式网络服务器--Address模块

Address模块

Linux使用Berkeley套接字接口进行网络编程,目前基本所有的系统上都支持。Berkeley套接字大部分API都需要传入一个地址作为参数,用于表示网络中的一台主机的通信地址。但是不同的协议类型对应的网络地址类型不一样。比如IPv4协议对应IPv4地址,其长度为32位,而IPv6协议对应IPv6地址,其长度为128

如果针对每种类型的地址都制定一套对应的API,那么套接字的API数量就会相对宠大,这对开发和维护都没好处。我们希望的是只使用一套通用的API就能实现各种地址类型的操作,比如针对IPv4地址的代码,能够在不修改或尽量少修改的前提下,就可用于IPv6地址。对此,Berkeley套接字接口拟定了一个通用套接字地址结构sockaddr,用于表示任意类型的地址,所有的套接字API在传入地址参数时都只需要传入sockaddr类型,以保证接口的通用性。除通用地址结构sockaddr外,还有一系列表示具体的网络地址的结构,这些具体的网络地址结构用于用户赋值,但在使用时,都要转化成sockaddr的形式

sockaddr表示通用套接字地址结构,其定义如下:

struct sockaddr {
    unsigned short sa_family; 	// 地址族(地址类型)
    char sa_data[14];         	// 地址内容
};

sockaddr_in表示IPv4地址结构,其定义如下:

struct sockaddr_in {
    unsigned short sin_family; 		// 地址族,IPv4的地址族为AF_INET
    unsigned short sin_port;   		// 端口
    struct in_addr sin_addr;   		// IP地址,IPv4的地址用一个32位整数来表示
    char sin_zero[8];          		// 填充位,填零即可
};

sockaddr_in6表示IPv6地址结构,其定义如下:

struct sockaddr_in6 {
    unsigned short sin6_family; // 地址族,IPv6的地址族为AF_INET6
    in_port_t sin6_port;        // 端口
    uint32_t sin6_flowinfo;     // IPv6流控信息
    struct in6_addr sin6_addr;  // IPv6地址,实际为一个128位的结构体
    uint32_t sin6_scope_id;     // IPv6 scope-id
};

通过上面的定义也可以发现,除sockaddr_in可以无缝转换为sockaddr外,sockaddr_in6sockaddr_un都不能转换为sockaddr,因为大小不一样

但这并不影响套接字接口的通用性,Berkeley套接字接口都是以指针形式接收sockaddr参数,并且额外需要一个地址长度参数,又由于以上所有的地址结构的前两个字节都表示地址族,所以通过sockaddr指针总能拿到传入地址的地址类型,通过地址类型判断出地址长度后,再通过sockaddr指针取适合该地址的长度即可拿到地址内容。比如上面的地址内容占14字节,这并不足以容纳一个12816字节的IPv6地址。但当以指针形式传入时,完全可以通过指针取到适合IPv6的长度

github

https://github.com/huxiaohei/tiger.git

实现

tiger采用Berkeley同样的设计思路,将IPv4IPv6网络地址和Unix本地地址,以及与地址先关的基本操作封装在Address模块中,方便后续对地址的使用和维护

类图

Address
static Address::ptr Create(const sockaddr *addr)
static bool Lookup(std::vector &r, const std::string &host, int family = AF_INET, int type = 0, int protocol = 0)
static Address::ptr LookupAny(const std::string &host, int family = AF_INET, int type = 0, int protocol = 0)
virtual ~Address()
virtual const sockaddr *get_addr()
virtual sockaddr *get_addr()
virtual const socklen_t get_addr_len()
virtual std::ostream &insert(std::ostream &os)
IPAddress
static IPAddress::ptr Create(const char *address, uint16_t port = 0)
static IPAddress::ptr LookupAny(const std::string &host, int family = AF_INET, int type = 0, int protocol = 0)
static bool InterfaceAddresses(std::multimap> &r, int family = AF_INET)
static bool InterfaceAddresses(std::vector> &r, const std::string &iface, int family = AF_INET)
virtual IPAddress::ptr broadcast_address(uint32_t prefix_len)
virtual IPAddress::ptr network_address(uint32_t prefix_len)
virtual IPAddress::ptr subnet_mask(uint32_t prefix_len)
UnixAddress
UnixAddress()
UnixAddress(const std::string &path)
IPv4Address
sockaddr_in m_addr
IPv6Address
sockaddr_in6 m_addr
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

虎小黑

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值