Unix Domain Socket和eventfd

news2025/3/7 2:18:24

在Linux开发中,Unix Domain Socketeventfd是两种不同的通信机制,它们的设计目标和适用场景有显著差异。以下分点解释并配合示例说明:


一、Unix Domain Socket(UDS)

1. 是什么?
  • 一种**本地进程间通信(IPC)**机制,基于文件系统路径而非网络端口
  • 协议族为AF_UNIX(而非网络Socket的AF_INET/AF_INET6
2. 核心特点
  • 高性能:数据直接在内核缓冲区传递,无需经过网络协议栈
  • 支持多种通信模式:流式(SOCK_STREAM,类似TCP)、数据报(SOCK_DGRAM,类似UDP)和可靠数据报(SOCK_SEQPACKET
  • 权限控制:通过文件系统权限(chmod)控制访问
  • 传递文件描述符:可通过sendmsg发送文件描述符(其他IPC机制难以实现)
3. 典型场景
  • Docker守护进程/var/run/docker.sock
  • 数据库本地连接:MySQL客户端通过/var/run/mysqld/mysqld.sock连接本地服务
  • X Window系统:GUI应用与X服务器的通信
  • Systemd服务:通过/run/systemd/下的socket文件通信
4. 代码片段
// 服务端
struct sockaddr_un addr = {.sun_family = AF_UNIX};
strcpy(addr.sun_path, "/tmp/mysocket");
bind(sock, (struct sockaddr*)&addr, sizeof(addr));

二、eventfd

1. 是什么?
  • 一个轻量级事件通知机制,本质是一个内核维护的64位计数器
  • 通过eventfd()系统调用创建,返回一个文件描述符
2. 核心特点
  • 极低开销:适合高频事件通知(如线程间唤醒)
  • 与epoll集成:可直接加入事件循环监听
  • 原子操作:读/写操作是原子的,线程安全
  • 跨进程可用:通过fork或传递fd给其他进程
3. 典型场景
  • 线程池唤醒:主线程通过写eventfd通知工作线程有新任务
  • 协程调度:如Golang的runtime用类似机制唤醒阻塞的goroutine
  • 信号处理:将信号转换为eventfd事件,避免在信号处理函数中执行复杂逻辑
  • IO多路复用集成:与epoll结合实现高效事件驱动架构
4. 代码片段
int efd = eventfd(0, EFD_NONBLOCK); // 创建eventfd
write(efd, &(uint64_t){1}, sizeof(uint64_t)); // 发送事件
read(efd, &counter, sizeof(uint64_t)); // 接收事件

三、Unix Socket vs TCP/UDP Socket

特性Unix Domain SocketTCP/UDP Socket
通信范围同一主机跨网络
地址形式文件路径(如/tmp/foo.sockIP地址+端口(如127.0.0.1:8080
协议开销无网络协议栈处理需要处理TCP握手/重传等
性能比TCP快2倍以上(实测)受网络延迟和带宽限制
权限控制文件系统权限依赖网络防火墙
数据边界保留流式/数据报均保留TCP是流式无边界,UDP有边界
示例对比
  1. Redis客户端连接

    • 远程连接:redis-cli -h 192.168.1.100 -p 6379(TCP)
    • 本地连接:redis-cli -s /tmp/redis.sock(UDS)
  2. Nginx本地配置

    server {
        listen 80;                  # TCP Socket
        listen unix:/var/run/nginx.sock; # Unix Socket
    }
    

四、组合使用案例

场景:构建一个高性能本地服务

  1. 使用Unix Socket处理客户端请求(高吞吐)
  2. 使用eventfd在服务内部线程间传递事件(低延迟)
  3. 用epoll同时监听UDS和eventfd,实现统一事件循环
// 伪代码示例
int uds_fd = create_unix_socket("/tmp/service.sock");
int efd = eventfd(0, EFD_NONBLOCK);

struct epoll_event events[2];
epoll_ctl(epfd, EPOLL_CTL_ADD, uds_fd, &(struct epoll_event){.events=EPOLLIN});
epoll_ctl(epfd, EPOLL_CTL_ADD, efd, &(struct epoll_event){.events=EPOLLIN});

while(1) {
    int n = epoll_wait(epfd, events, 2, -1);
    for (每个就绪事件) {
        if (事件来自uds_fd) 处理客户端请求();
        if (事件来自efd) 处理内部事件();
    }
}

总结选择依据:

  • 需要跨主机通信 → TCP/UDP Socket
  • 本地高性能IPC → Unix Domain Socket
  • 轻量级事件通知 → eventfd
  • 混合场景 → UDS + eventfd + epoll组合

在 Linux 开发中,Socket Pair 是一种特殊的进程间通信(IPC)机制,与 Unix Domain Socket (UDS)TCP/UDP Socketeventfd 有显著区别。以下是它们的核心差异、使用场景和实际示例:


1. Socket Pair 是什么?

  • 定义:通过 socketpair() 系统调用创建的一对已连接的匿名 Unix Domain Socket
  • 特点
    • 无需绑定文件路径:直接创建一对已连接的 socket,没有文件系统路径。
    • 双向通信:两个 socket 可以互相读写(类似全双工管道)。
    • 匿名性:仅限同一进程或父子进程间使用(无法被无关进程访问)。
    • 支持多种模式:流式(SOCK_STREAM)或数据报(SOCK_DGRAM)。
示例代码
#include <sys/socket.h>
#include <stdio.h>

int main() {
    int sock_pair[2];
    socketpair(AF_UNIX, SOCK_STREAM, 0, sock_pair); // 创建 socket pair

    char buf[128];
    write(sock_pair[0], "Hello from sock0", 16);       // 写入 sock0
    read(sock_pair[1], buf, sizeof(buf));            // 从 sock1 读取
    printf("Received: %s\n", buf);                   // 输出 "Hello from sock0"

    close(sock_pair[0]);
    close(sock_pair[1]);
    return 0;
}

2. Socket Pair 与其他通信机制的区别

对比 Unix Domain Socket (UDS)
特性Socket PairUnix Domain Socket
地址绑定匿名,无需文件路径需绑定文件路径(如 /tmp/foo.sock
可见性仅限创建进程及其子进程所有有权限访问文件的进程
使用场景父子进程或线程间通信任意进程间通信(需共享路径)
性能更高(无需文件系统操作)高(但需文件路径管理)
对比 TCP/UDP Socket
特性Socket PairTCP/UDP Socket
通信范围同一主机,同一进程或父子进程跨网络或本机(需网络协议栈)
地址形式无地址(匿名)IP地址 + 端口
协议开销无网络协议栈开销需处理 TCP 握手、UDP 不可靠性等
对比 eventfd
特性Socket Paireventfd
通信方向双向(支持数据收发)单向(仅事件通知,无数据内容)
数据类型任意字节流或数据报64位计数器(仅数值增减)
典型用途进程/线程间数据交换事件通知(如唤醒线程池)

3. Socket Pair 的典型使用场景

场景 1:父子进程通信

父进程创建 socket pair,fork 子进程后,父子各关闭一个 socket,实现双向通信:

int sock_pair[2];
socketpair(AF_UNIX, SOCK_STREAM, 0, sock_pair);

if (fork() == 0) {  // 子进程
    close(sock_pair[0]);  // 关闭父进程端
    write(sock_pair[1], "Child message", 13);
} else {            // 父进程
    close(sock_pair[1]);  // 关闭子进程端
    read(sock_pair[0], buf, sizeof(buf));
}
场景 2:线程间通信

多个线程共享同一 socket pair,实现高效数据传递(需注意线程同步)。

场景 3:替代管道(Pipe)

Socket pair 比传统管道更灵活:

  • 管道是单向的,socket pair 是双向的。
  • Socket pair 支持数据报模式(SOCK_DGRAM)。

4. 组合使用案例

需求:构建一个多进程服务,主进程通过 socket pair 与子进程通信,同时使用 eventfd 通知事件。

// 主进程
int sock_pair[2];
socketpair(AF_UNIX, SOCK_STREAM, 0, sock_pair);

int efd = eventfd(0, EFD_NONBLOCK);

if (fork() == 0) {  // 子进程
    close(sock_pair[0]);
    // 监听 sock_pair[1] 和 efd
    while (1) {
        // 通过 sock_pair[1] 与主进程交换数据
        // 通过 eventfd 接收事件通知
    }
} else {            // 主进程
    close(sock_pair[1]);
    write(sock_pair[0], "Task data", 9);  // 发送数据
    write(efd, &(uint64_t){1}, 8);       // 发送事件
}

5. 总结:如何选择?

  • 需要跨网络通信TCP/UDP Socket
  • 本地任意进程间通信Unix Domain Socket(需文件路径)。
  • 父子进程或线程间高效双向通信Socket Pair(匿名,无需文件路径)。
  • 轻量级事件通知eventfd(无数据内容,仅计数器)。
  • 混合场景 → 组合使用(如 Socket Pair + eventfd + epoll)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2310818.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【万字长文】基于大模型的数据合成(增强)及标注

写在前面 由于合成数据目前是一个热门的研究方向&#xff0c;越来越多的研究者开始通过大模型合成数据来丰富训练集&#xff0c;为了能够从一个系统的角度去理解这个方向和目前的研究方法便写了这篇播客&#xff0c;希望能对这个领域感兴趣的同学有帮助&#xff01; 欢迎点赞&…

MacBook上API调⽤⼯具推荐

在当今的软件开发中&#xff0c;API调用工具已经成为了开发者不可或缺的助手。无论是前端、后端还是全栈开发&#xff0c;API的调试、测试和管理都是日常工作中的重要环节。想象一下&#xff0c;如果没有这些工具&#xff0c;开发者可能需要手动编写复杂的CURL命令&#xff0c;…

【数据结构】LRUCache|并查集

目录 一、LRUCache 1.概念 2.实现:哈希表双向链表 3.JDK中类似LRUCahe的数据结构LinkedHashMap &#x1f525;4.OJ练习 二、并查集 1. 并查集原理 2.并查集代码实现 3.并查集OJ 一、LRUCache 1.概念 最近最少使用的&#xff0c;一直Cache替换算法 LRU是Least Recent…

初识Qt · 信号与槽 · 基础知识

目录 前言&#xff1a; 信号和槽初识 两个问题 前言&#xff1a; 本文我们正式开始介绍信号与槽这个概念&#xff0c;在谈及Qt中的信号与槽这个概念之前&#xff0c;我们不妨回顾一下Linux中的信号&#xff0c;比如发生了除0错误&#xff0c;OS就会给该进程发送一个信号&am…

Java高频面试之集合-03

hello啊&#xff0c;各位观众姥爷们&#xff01;&#xff01;&#xff01;本baby今天来报道了&#xff01;哈哈哈哈哈嗝&#x1f436; 面试官&#xff1a;说说ArrayList和LinkedList的区别 ArrayList 与 LinkedList 的详细对比 一、底层数据结构 特性ArrayListLinkedList存…

宇树科技再落一子!天羿科技落地深圳,加速机器人创世纪

2025年3月5日&#xff0c;机器人行业龙头宇树科技&#xff08;Unitree&#xff09;在深圳再添新动作——全资子公司深圳天羿科技有限公司正式成立。这家注册资本10万元、法定代表人周昌慧的新公司&#xff0c;聚焦智能机器人研发与销售&#xff0c;标志着宇树科技在华南市场的战…

【长安大学】苹果手机/平板自动连接认证CHD-WIFI脚本(快捷指令)

背景&#xff1a; 已经用这个脚本的记得设置Wifi时候&#xff0c;关闭“自动登录” 前几天实在忍受不了CHD-WIFI动不动就断开&#xff0c;一天要重新连接&#xff0c;点登陆好几次。试了下在网上搜有没有CHD-WIFI的自动连接WIFI自动认证脚本&#xff0c;那样我就可以解放双手&…

计算机毕业设计SpringBoot+Vue.js电商平台(源码+文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

【杂谈】信创电脑华为w515(统信系统)登录锁定及忘记密码处理

华为w515麒麟芯片版&#xff0c;还有非麒麟芯片版本&#xff0c;是一款信创电脑&#xff0c;一般安装的UOS系统。 准备一个空U盘&#xff0c;先下载镜像文件及启动盘制作工具&#xff0c;连接如下&#xff1a; 百度网盘 请输入提取码 http://livecd.uostools.com/img/apps/l…

初始提示词(Prompting)

理解LLM架构 在自然语言处理领域&#xff0c;LLM&#xff08;Large Memory Language Model&#xff0c;大型记忆语言模型&#xff09;架构代表了最前沿的技术。它结合了存储和检索外部知识的能力以及大规模语言模型的强大实力。 LLM架构由外部记忆模块、注意力机制和语…

Vue+el-upload配置minIO实现大文件的切片并发上传、上传进度展示、失败重试功能

vue3el-upload实现切片上传 效果图 初始界面 上传中的界面 上传完成的界面 上传失败的界面 <template><div><el-uploadclass"BigFileUpload"ref"uploadRef"action"#"drag:show-file-list"false":on-change"…

正则表达式梳理(基于python)

正则表达式&#xff08;regular expression&#xff09;是一种针对字符串匹配查找所定义的规则模式&#xff0c;独立于语言&#xff0c;但不同语言在实现上也会存在一些细微差别&#xff0c;下面基于python对常用的相关内容进行梳理。 文章目录 一、通用常识1.通配符ps.反义 2.…

【仿muduo库one thread one loop式并发服务器实现】

文章目录 一、项目介绍1-1、项目总体简介1-2、项目开发环境1-3、项目核心技术1-4、项目开发流程1-5、项目如何使用 二、框架设计2-1、功能模块划分2-1-1、SERVER模块2-1-2、协议模块 2-2、项目蓝图2-2-1、整体图2-2-2、模块关系图2-2-2-1、Connection 模块关系图2-2-2-2、Accep…

服务流程设计和服务或端口重定向及其websocket等应用示例

服务流程设计和服务或端口重定向及其websocket等应用示例 目录 服务或端口重定向的服务设计和websocket等应用示例 一、通用请求控制流程 1.1、入口 1.2、所有GET请求首先预检控制单元 1.3、http请求会分别自动307重定向 1.4、所有请求首先执行跨源控制单元 1.5、然后…

【数据库】关系代数

关系代数 一、关系代数的概念二、关系代数的运算2.1 并、差、交2.2 投影、选择2.3 笛卡尔积2.4 连接2.5 重命名2.6 优先级 一、关系代数的概念 关系代数是一种抽象的数据查询语言用对关系的运算来表达查询 运算对象&#xff1a;关系运算符&#xff1a;4类运算结果&#xff1a;…

ubuntu20 安装python2

1. 确保启用了 Universe 仓库 在某些情况下&#xff0c;python2-minimal 包可能位于 Universe 仓库中。你可以通过以下命令启用 Universe 仓库并更新软件包列表&#xff1a; bash复制 sudo add-apt-repository universe sudo apt update 然后尝试安装&#xff1a; bash复制…

MySQL无法连接到本地localhost的解决办法2024.11.8

问题描述&#xff1a;我的MySQL可以远程连接服务器&#xff0c;但无法连接自己的localhost。 错误提示&#xff1a; 2003 - Cant connet to MySQL server on localhost(10061 "Unknown error")查找问题原因&#xff1a; 1. 检查环境变量是否正确&#xff1a;发现没…

最新Spring Security实战教程(一)初识Spring Security安全框架

&#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Micro麦可乐的博客 &#x1f425;《Docker实操教程》专栏以最新的Centos版本为基础进行Docker实操教程&#xff0c;入门到实战 &#x1f33a;《RabbitMQ》…

告别GitHub连不上!一分钟快速访问方案

一、当GitHub抽风时&#xff0c;你是否也这样崩溃过&#xff1f; &#x1f621; npm install卡在node-sass半小时不动&#x1f62d; git clone到90%突然fatal: early EOF&#x1f92c; 改了半天hosts文件&#xff0c;第二天又失效了... 根本原因&#xff1a;传统代理需要复杂…

Leetcode 1477. 找两个和为目标值且不重叠的子数组 前缀和+DP

原题链接&#xff1a; Leetcode 1477. 找两个和为目标值且不重叠的子数组 class Solution { public:int minSumOfLengths(vector<int>& arr, int target) {int narr.size();int sum0;int maxnINT_MAX;vector<int> dp(n,maxn);//dp[i]表示以索引i之前的满足要求…