《socket 编程必学:sockaddr_in 与 sockaddr 的前世今生》

1 结构体定义

1.1 sockaddr_in

1.sockaddr_in 是 Linux/Unix 网络编程中用于表示 IPv4 地址 的核心数据结构,它是 sockaddr 结构体的具体实现,专门用于 IPv4 网络通信。
2.当你创建好套接字socket并要去用bind()函数绑定它的地址和端口信息时,此时就要初始化sockaddr_in结构体,一般使用

#include <netinet/in.h>

struct sockaddr_in {
    sa_family_t    sin_family;   // 地址族 (Address Family)
    in_port_t      sin_port;     // 端口号 (Port number)
    struct in_addr sin_addr;     // IPv4地址 (32-bit)
    unsigned char  sin_zero[8];  // 填充字段 (未使用)
};

struct in_addr {
    in_addr_t s_addr;           // 32位IPv4地址 (网络字节序)
};

初始化实例

//绑定前先填充地址结构体
struct sockaddr_in local;
memset(&local, 0, sizeof(local));// 清空结构体
local.sin_family = AF_INET;// IPv4地址族
local.sin_port = ::htons(8080); // 要被发送给对方的,即要发到网络中!
                                //端口8080(主机→网络字节序)
local.sin_addr.s_addr = inet_addr("127.0.0.1"); // IP地址转换
//或者也可以 local.sin_addr.s_addr = INADDR_ANY;
//INADDR_ANY:表示"任意可用的网络接口"或"所有本地IP地址"。

1.2 sockaddr

sockaddr 是 Linux/Unix 网络编程中最基础的通用套接字地址结构,用于表示各种类型的网络地址。它是一个通用的地址容器,实际使用时会被转换为更具体的地址结构(如 sockaddr_in 或 sockaddr_in6)。

#include <sys/socket.h>

struct sockaddr {
    sa_family_t sa_family;  // 地址族(Address Family)
    char        sa_data[14]; // 协议特定地址信息
};

sockaddr 的主要作用是作为通用参数类型,用于各种套接字函数(如 bind(), connect(), accept() 等)的地址参数。实际使用时需要转换为具体的地址结构体,所以我们一般都使用sockaddr_in

2 字节序转换

头文件 <arpa/inet.h>

函数作用实例
htons()主机字节序→网络字节序(16位)htons(8080)
ntohs()网络字节序→主机字节序(16位)ntohs(server_addr.sin_port)
hton1()主机字节序→网络字节序(32位)htonl(INADDR_ANY)
ntohl()网络字节序→主机字节序(32位)ntohl(addr.s_addr)
#include <arpa/inet.h>

uint16_t host_port = 8080;
uint32_t host_ip = 0xC0A80101; // 192.168.1.1

// 转换为网络字节序
uint16_t net_port = htons(host_port); 
uint32_t net_ip = htonl(host_ip);

// 转换回主机字节序
uint16_t port_back = ntohs(net_port);
uint32_t ip_back = ntohl(net_ip);

h 	- host 主机,主机字节序
to 	- 转换成什么
n 	- network 网络字节序
s 	- short : unsigned short   无符号短整型,两个字节
l 	- long : unsigned int		 无符号长整型,四个字节

3 IP地址转换

// p:点分十进制的IP字符串,n:表示network,网络字节序的整数
int inet_pton(int af, const char *src, void *dst);
// 将网络字节序的整数,转换成点分十进制的IP地址字符串
const char *inet_ntop(int af, const void *src, char *dst, socklen_t size);
	af:地址族: AF_INET AF_INET6
	src: 要转换的ip的整数的地址
	dst: 转换成IP地址字符串保存的地方
	size:第三个参数的大小(数组的大小)
	返回值:返回转换后的数据的地址(字符串),和 dst 是一样的
函数作用实例
inet_pton()字符串 → 二进制(支持IPv4/IPv6)inet_pton(AF_INET, “192.168.1.1”, &addr.s_addr);
inet_ntop()二进制 → 字符串inet_ntop(AF_INET, &addr, str, INET_ADDRSTRLEN);
inet_addr()IPv4 点分十进制字符串地址转换为网络字节序 32 位整数inet_addr(“192.168.1.1”)
// IPv4示例
struct in_addr addr;
inet_pton(AF_INET, "192.168.1.1", &addr.s_addr);
    
char str[INET_ADDRSTRLEN];
const char* result = inet_ntop(AF_INET, &addr, str, INET_ADDRSTRLEN);

const char *ip_str = "192.168.1.1";
in_addr_t ip_addr = inet_addr(ip_str);

// IPv6示例
struct in6_addr ipv6;
if (inet_pton(AF_INET6, "2001:db8::1", &ipv6) != 1) {
    // 处理错误
    
char ipv6_str[INET6_ADDRSTRLEN];

inet_ntop(AF_INET6, &ipv6, ipv6_str, INET6_ADDRSTRLEN);
}

4 封装一个inetaddr类

主要功能:处理 IPv4 的套接字地址信息,简化了 sockaddr_in 的手动管理,适合封装在更高级的网络库中。

#pragma once

#include <iostream>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "Common.hpp"
#define CONV(v) (struct sockaddr *)(v)//宏定义,强制类型转换。
class InetAddr
{
private:
    void PortNet2Host() // 将端口号从网络字节序转为主机字节序
    {
        _port = ::ntohs(_net_addr.sin_port);
    }
    void IpNet2Host() // 将 IP 地址从网络字节序转为主机字节序的可读字符串
    {
        char ipbuffer[64];
        const char *ip = ::inet_ntop(AF_INET, &_net_addr.sin_addr, ipbuffer, sizeof(ipbuffer));
        (void)ip;
    }

public:
    // 封装了 sockaddr_in 结构体,提供更友好的接口访问 IP 和端口。
    InetAddr()
    {
    }
    InetAddr(const struct sockaddr_in &addr) : _net_addr(addr)
    {
        PortNet2Host();
        IpNet2Host();
    }
    InetAddr(uint16_t port) : _port(port), _ip("")
    {
        _net_addr.sin_family = AF_INET;
        _net_addr.sin_port = htons(_port);
        _net_addr.sin_addr.s_addr = INADDR_ANY;
    }
    struct sockaddr *NetAddr() { return CONV(&_net_addr); }
    socklen_t NetAddrLen() { return sizeof(_net_addr); }
    std::string Ip() { return _ip; }
    uint16_t Port() { return _port; }
    ~InetAddr()
    {
    }

private:
    struct sockaddr_in _net_addr;
    std::string _ip;
    uint16_t _port;
};

4.1 构造 InetAddr 对象

//默认构造(空地址)
InetAddr addr;  // 默认 IP="0.0.0.0", Port=0

//指定端口(自动绑定任意 IP)
InetAddr addr(8080);  // IP="0.0.0.0"(INADDR_ANY),Port=8080(主机字节序)

//从 sockaddr_in 构造
struct sockaddr_in peer;
// ...(填充 sock_addr,如 accept() 返回的客户端地址)
InetAddr cli(peer);  // 自动转换网络字节序
std::string clientinfo = cli.Ip() + ":" + std::to_string(cli.Port()) + " # " + inbuffer;

4.2 获取地址信息

InetAddr addr(8080); 

// 获取 IP(字符串形式,如 "192.168.1.100")
std::string ip = addr.IP();

// 获取端口(主机字节序,如 8080)
uint16_t port = addr.Port();

// 获取底层 sockaddr*(用于 bind/connect 等)
struct sockaddr* sa = addr.NetAddr();
socklen_t len = addr.NetAddrLen();

//bind
int sockfd = socket(AF_INET, SOCK_STREAM, 0);
// 绑定 0.0.0.0:8080
InetAddr addr(8080);
bind(sockfd, addr.NetAddr(), addr.NetAddrLen());

如有错误,欢迎指正!

借鉴文章:
https://blog.csdn.net/weixin_43412762/article/details/136418447

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值