在 Linux 中,有些特殊文件(如 /proc/*/environ、/proc/*/cmdline、某些二进制配置文件)的内容不是用换行符 \n 分隔,而是用 null 字节 \0 分隔。这导致直接用 cat 看起来是空的,用 grep、awk 等常规工具处理时会出错或显示不全。本文将介绍一系列处理这类"null-separated"内容的实用小技巧。
1. 最常用的三种查看方式
bash
# 方法1:tr 替换 null 为换行(最直观)
tr '\0' '\n' < /proc/self/environ
# 方法2:xargs -0(专为 null 分隔设计)
xargs -0 -n1 < /proc/self/environ
# 方法3:strings(自动提取可打印字符串)
strings /proc/self/environ
三者效果类似,但细节有区别:
tr '\0' '\n':完整保留所有内容,包括空字符串。xargs -0 -n1:每条单独输出一行,自动跳过连续多个 null。strings:只输出可打印的 ASCII 字符串,过滤掉二进制垃圾。
2. 进阶组合技巧
排序、去重、统计
bash
# 排序查看环境变量
tr '\0' '\n' < /proc/self/environ | sort
# 去重(如果有重复变量)
tr '\0' '\n' < /proc/self/environ | sort -u
# 统计有多少个环境变量
xargs -0 -n1 < /proc/self/environ | wc -l
搜索特定变量
bash
# 查找包含 PATH 的变量
tr '\0' '\n' < /proc/self/environ | grep PATH
# 或用 xargs + grep
xargs -0 -n1 < /proc/self/environ | grep -i display
# 只看变量名(去掉 = 后面的值)
xargs -0 -n1 < /proc/self/environ | cut -d= -f1
同时查看多个进程的环境
bash
echo "=== PID 1 (systemd) ==="
tr '\0' '\n' < /proc/1/environ | sort
echo "=== 当前 shell ==="
tr '\0' '\n' < /proc/self/environ | sort | head -20
echo "=== sshd 进程(假设 PID 1234)==="
tr '\0' '\n' < /proc/1234/environ | grep SSH
3. 处理 /proc/*/cmdline 的经典用法
/proc/*/cmdline 用 null 分隔命令行参数。
bash
# 查看某个进程的完整命令行(参数间有空格也能正确显示)
tr '\0' '\n' < /proc/1234/cmdline | paste -sd " "
# 或者更常见:
xargs -0 < /proc/1234/cmdline
# 列出所有进程的命令行(简化版 ps)
for pid in /proc/[0-9]*; do
echo -n "$(basename $pid): "
tr '\0' ' ' < $pid/cmdline 2>/dev/null || echo
done
4. 其他 null-separated 场景的小技巧
find + -print0 配合 xargs -0(避免文件名含空格/换行问题)
bash
# 安全处理文件名含空格、换行的场景
find /path -name "*.txt" -print0 | xargs -0 rm -v
find /path -type f -print0 | xargs -0 ls -l
while read + xargs -0 循环处理
bash
xargs -0 -n1 < /proc/self/environ | while read line; do
echo "变量: $line"
done
一行命令导出当前环境到文件(保留 null 分隔)
bash
# 备份当前环境
cp /proc/self/environ myenv.backup
# 恢复(危险,仅演示)
xargs -0 -n1 < myenv.backup | xargs -0 export