创建套接字并bind的详细过程

bind()套接字编程 中的一个关键函数,用于 将套接字(socket)与特定的 IP 地址和端口号绑定 。它通常用于 服务器端,但也可以用于客户端(如指定本地端口)。


1. bind() 的作用

  • 服务器端:绑定一个固定的 IP 和端口,以便客户端可以连接。

  • 客户端(可选):可以绑定特定的本地端口(如某些协议要求固定端口)。

关键点:

  • TCP/UDP 服务器必须调用 bind(),否则客户端无法连接。

  • 客户端通常不需要 bind(),系统会自动分配临时端口。


2. bind() 的函数原型

cpp 复制代码
#include <sys/socket.h>

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

参数说明:

参数 说明
sockfd 套接字文件描述符(由 socket() 创建)
addr 指向 struct sockaddr 的指针,包含 IP 和端口
addrlen addr 结构体的大小(sizeof(struct sockaddr_in)

返回值:

  • 成功 :返回 0

  • 失败 :返回 -1,并设置 errno(如 EADDRINUSE 表示端口被占用)


3. bind() 的使用步骤

(1) 创建套接字
cpp 复制代码
int sockfd = socket(AF_INET, SOCK_STREAM, 0); // TCP
// 或
int sockfd = socket(AF_INET, SOCK_DGRAM, 0);  // UDP
(2) 设置地址结构(struct sockaddr_in
cpp 复制代码
struct sockaddr_in addr;
memset(&addr, 0, sizeof(addr)); // 清空结构体
addr.sin_family = AF_INET;       // IPv4
addr.sin_addr.s_addr = htonl(INADDR_ANY); // 监听所有网卡(0.0.0.0)
addr.sin_port = htons(8080);     // 绑定端口 8080

关键字段说明:

字段 说明
sin_family 地址族(AF_INET 表示 IPv4)
sin_addr.s_addr IP 地址(INADDR_ANY 表示监听所有网卡)
sin_port 端口号(必须用 htons() 转换字节序)
(3) 调用 bind()
cpp 复制代码
if (bind(sockfd, (struct sockaddr*)&addr, sizeof(addr)) < 0) {
    perror("bind failed");
    exit(1);
}

4. bind() 的常见问题

(1) 端口被占用(EADDRINUSE
  • 原因:另一个进程已经绑定了该端口。

  • 解决方法

    • 换一个端口。

    • 使用 SO_REUSEADDR 选项允许端口复用:

      cpp 复制代码
      int opt = 1;
      setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
(2) 权限不足(EACCES
  • 原因 :绑定 1024 以下的端口 (如 80、443)需要 root 权限。

  • 解决方法

    • sudo 运行程序。

    • 改用 1024 以上的端口(如 8080)。

(3) 绑定失败(EINVAL
  • 原因 :套接字已经绑定过,或者 addr 结构体不正确。

  • 解决方法:检查代码逻辑。


5. bind() 在 TCP/UDP 中的区别

TCP UDP
是否需要 bind() 服务器必须,客户端可选 服务器必须,客户端可选
典型用途 让客户端能 connect() 让客户端能 sendto()
绑定后操作 listen() + accept() 直接 recvfrom()/sendto()

6. 完整示例(TCP 服务器)

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>

int main() {
    // 1. 创建套接字
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) {
        perror("socket failed");
        exit(1);
    }

    // 2. 设置地址结构
    struct sockaddr_in addr;
    memset(&addr, 0, sizeof(addr));
    addr.sin_family = AF_INET;
    addr.sin_addr.s_addr = htonl(INADDR_ANY); // 0.0.0.0
    addr.sin_port = htons(8080); // 绑定 8080 端口

    // 3. 绑定
    if (bind(sockfd, (struct sockaddr*)&addr, sizeof(addr)) < 0) {
        perror("bind failed");
        exit(1);
    }

    printf("Server is running on port 8080...\n");
    close(sockfd);
    return 0;
}

总结

  • bind() 的作用 :将套接字绑定到 IP + 端口

  • 服务器必须 bind(),客户端通常不需要。

  • 常见错误

    • EADDRINUSE(端口被占用)→ 使用 SO_REUSEADDR

    • EACCES(权限不足)→ 改用高端口或 sudo

  • TCP vs UDPbind() 的用法基本相同,但后续操作不同(listen() vs recvfrom())。

相关推荐
雅欣鱼子酱18 小时前
USB Type-C PD取电(诱骗,诱电,SINK),筋膜枪专用取电芯片
网络·人工智能·芯片·电子元器件
AI大模型应用之禅1 天前
全球股市估值与可持续农业垂直种植技术的关系
网络·ai
掘根1 天前
【仿Muduo库项目】HTTP模块2——HttpRequest子模块,HttpResponse子模块
网络·网络协议·http
MOON404☾1 天前
006.Backdoor后门编写
网络·安全·网络安全·系统安全
ha20428941941 天前
Linux操作系统学习记录之---TcpSocket
linux·网络·c++·学习
李白你好1 天前
Burp Suite插件 | 高级HTTP头操作工具
网络·网络协议·http
石像鬼₧魂石1 天前
22端口(OpenSSH 4.7p1)渗透测试完整复习流程(含实战排错)
大数据·网络·学习·安全·ubuntu
xwj_8655743321 天前
HTTP协议、接口测试流程、接口测试流程&文档&用例
网络·网络协议·http
RisunJan1 天前
Linux命令-iptables-restore命令(从文件批量恢复 iptables 防火墙规则)
linux·运维·网络
..过云雨1 天前
网络计算器实现 - 自定义套接字+序列化+守护进程
网络·网络协议·tcp/ip