sed 原地替换文件时遇到的趣事

哈喽大家好,我是咸鱼

在文章《三剑客之 sed》中咸鱼向大家介绍了文本三剑客中的 sed

sed 全名叫 stream editor,流编辑器,用程序的方式来编辑文本

那么今天咸鱼打算讲一下我在用 sed 原地替换文件时遇到的趣事

sed 让文件属性变了?

有这么一个普通文件 test.txt ,内容如下:

csharp 复制代码
[root@localhost /opt]# cat test.txt 
My name is Ammon
hello world!
hahahahaha

link_test.txt 是一个软链接(Symbolic Link,也可以叫符号链接)文件,指向 test.txt 文件

软链接文件类似于 Windows 的快捷方式,它实际上是一个特殊的文件。关于软链接相关的内容这里不过多介绍

如果我们对软链接文件内容进行修改,源文件是会跟着改变的

当我使用 sed 对软链接文件link_test.txt 进行内容修改时,命令如下

arduino 复制代码
[root@localhost /opt]# sed -i 's/Ammon/Edsion/g' link_test.txt 

有趣的现象发生了:使用 sed 对软链接文件link_test.txt 进行内容修改之后,源文件test.txt内容没有变化,软链接文件link_test.txt 的内容发生变化且变成了普通文件

怎么样,是不是很神奇?sed 不但把文件内容改了,还把文件属性也给改了

我们用 strace 工具来查看一下在执行 sed 命令时操作系统做了哪些操作

strace 一个系统调用跟踪工具,它会跟踪并记录命令运行期间的系统调用和信号

arduino 复制代码
[root@localhost /opt]# strace sed -i 's/Ammon/Edsion/g' link_test.txt > strace.log 2>&1

具体内容有点多,我们着重看下关键内容

由上图可以知道,如果我们使用 sed 对一个文件进行原地替换,需要有一个临时文件,sed 先把修改内容写入到这个文件,最后将文件 rename 到原来的地方

那我的需求是对软链接文件 link_test.txt 进行内容修改,软链接文件属性不变,且源文件的内容也跟着改变,有没有什么方法呢?

也就是说我们希望命令在执行的时候能够解析出来软链接文件后面指向的源文件,直接对源文件进行读--处理--写,最后 rename

sed 命令有一个选项 --follow-symlinks

sql 复制代码
--follow-symlinks: 
    follow symlinks when processing in place

我们在执行的时候加上这个选项:

css 复制代码
[root@localhost /opt]# strace sed -i --follow-symlinks 's/Ammon/Edsion/g' link_test.txt > strace.log 2>&1

总结一下:

  • sed 替换的底层逻辑是把更新后的内容写入一个临时文件里面,然后再 rename 这个临时文件
  • 这样就会使得如果没有添加 --follow-symlinks 选项的话,对软链接文件进行 sed 操作就会使得软链文件变成了一个常规文件(其实是那个临时文件重命名导致的)
  • 如果要对软链接文件指向的源文件进行操作,就需要添加 --follow-symlinks 选项
相关推荐
青草地溪水旁35 分钟前
pthread_create详解:打开多线程编程的大门
linux·c/c++
Elastic 中国社区官方博客1 小时前
CI/CD 流水线与 agentic AI:如何创建自我纠正的 monorepos
大数据·运维·数据库·人工智能·搜索引擎·ci/cd·全文检索
A-刘晨阳2 小时前
Linux安装centos8及基础配置
linux·运维·服务器·操作系统·centos8
恒雨田2 小时前
解决 jenkins 用户 SSH 连接目标服务器时的 Permission denied 问题
运维·ssh·jenkins
不老刘2 小时前
macOS/Linux ClaudeCode 安装指南及 Claude Sonnet 4.5 介绍
linux·macos·ai编程·claude·vibecoding
迎風吹頭髮2 小时前
UNIX下C语言编程与实践14-UNIX 文件系统格式化:磁盘分区与文件系统创建原理
运维·c语言·unix
Saniffer_SH2 小时前
【高清视频】CXL 2.0 over Fibre演示和答疑 - 将内存拉到服务器10米之外
运维·服务器·网络·人工智能·驱动开发·计算机外设·硬件工程
野熊佩骑3 小时前
一文读懂Redis之数据持久化
linux·运维·数据库·redis·缓存·中间件·centos
Murphy_lx3 小时前
Linux(操作系统)文件系统--对打开文件的管理
linux·c语言·数据库
saber_andlibert4 小时前
【Linux】IPC——命名管道(fifo)
linux·运维·服务器