AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 779636
Accepted
Vlastimil Burián
Vlastimil Burián
Asked: 2024-07-07 03:47:18 +0800 CST2024-07-07 03:47:18 +0800 CST 2024-07-07 03:47:18 +0800 CST

如何以 POSIX 方式忽略“警告:命令替换:忽略输入中的空字节”?

  • 772

今天,我正在使用装有 Debian 12 Bookworm 的 Raspberry Pi 4,我发现一些系统文本文件很奇怪,我需要这两个:

/sys/firmware/devicetree/base/model
/sys/firmware/devicetree/base/serial-number

读取并存储在变量中bash如下:

rpi_model_name=$(cat /sys/firmware/devicetree/base/model)
rpi_serial_number=$(cat /sys/firmware/devicetree/base/serial-number)

产生以下警告,我无法通过重定向stderr来消除/dev/null例如:

warning: command substitution: ignored null byte in input

提到的重定向看起来像这个命令片段;请注意,我们仍然会收到令人讨厌的警告:

rpi_model_name=$(cat /sys/firmware/devicetree/base/model 2>/dev/null)
-bash: warning: command substitution: ignored null byte in input

我刚安装了两个 shell,bash和dash。虽然没有那个警告,但dash行为正常bash;似乎更拗口。所以,目前,我不确定其他 shell 是否也会发出这个或类似的警告。

我的目标是在所有 shell 中隐藏此警告。我不关心它的来源或值,我只希望它消失。

为了可移植性,解决方案必须以 POSIX 形式编写。

shell-script
  • 3 3 个回答
  • 25 Views

3 个回答

  • Voted
  1. Vlastimil Burián
    2024-07-07T03:47:18+08:002024-07-07T03:47:18+08:00

    这对我使用 Debian 12 Bookworm 的 Raspberry Pi 4 来说是有效的,它应该是一个可移植的解决方案(遵守 POSIX)。

    我们可以将所有 NULL 字符转换为空字符串,从而使用tr实用程序(POSIX 手册页)从字符串中有效地删除其出现的所有位置(我们知道在我的情况下,两种情况下都只有一个字符串和一个末尾的 NULL 字符,所以我们不会删除多个 NULL 字符,只删除字符串末尾的一个。)


    让我们以第一个提到的文件为例(/proc/device-tree/model按照Marcus Müller的回答阅读可能会更好):

    rpi_model_name=$(cat /proc/device-tree/model|tr -d '\0')
    

    甚至摆脱cat:

    rpi_model_name=$(tr -d '\0' </proc/device-tree/model)
    

    以下内容printf可以与 ,.. 互换echo,只要您觉得舒服即可:

    printf '%s\n' "$rpi_model_name"
    

    应该输出类似这样的内容:

    Raspberry Pi 4 Model B Rev 1.5
    

    这可能不是唯一的方法,请随意发布您自己的方法,或改进这个答案。

    • 1
  2. Marcus Müller
    2024-07-07T04:42:37+08:002024-07-07T04:42:37+08:00

    忽略或明确删除未知数量的零字节而不了解它们的来源似乎不是一个好主意!所以,让我们实际看看它们来自哪里,以及在什么情况下我们可能想要删除它。

    首先,为什么会出现错误?shell 字符串本身在内部是一个以零结尾的字符串(如果您熟悉该编程语言,则为“C 字符串”)。因此,它不能包含 0 字节;这始终表示字符串的结尾。因此,当您尝试在 bash(或任何其他 POSIX shell)中的字符串中存储带有零字节的内容时,该字符串会“清除”所有 0 字节,否则您会将整个输出保存到该字符串变量中,但无论何时读取它,您只会获得直到(但不包括)第一个0字节的所有字符。Bash 程序员知道这是错误的一大来源,所以他们会警告您。

    现在,我们可以安全地从输入中删除这些字节而不破坏它吗?

    根据内核文档,这是:

    每个文件的内容都是来自设备树的精确二进制数据。

    因此,设备树string类型属性以 0 字节结尾,而stringlist类型属性则是多个以 0 结尾的字符串,它们只是连接在一起。因此,在您的例子中,模型字段正确地以零字节结尾;这是精确二进制数据的一部分,因此也存在于 sysfs 条目中。

    但因为我们知道这是最后一个字节,

    rpi_model_name="$(head -c -1 /proc/device-tree/base/model)"
    

    可以工作。而且,额外的好处是,如果这实际上不是一个单独的字符串,而是多个连接在一起的字符串(例如,您会在compatible条目中看到这种情况!),您仍然会收到警告,而不是一些没有空格的乱七八糟的东西。

    当然,您可能希望更一般地处理这个问题:如果某个东西是字符串列表,您可能仍然希望读取它,然后按元素方式使用它(例如用于打印)或者将其连接起来,例如与换行符或空格连接。

    function read_devicetree_node() {
      mapfile -d '' result < "$1"
    }
    
    ## use like this
    
    read_devicetree_node /proc/device-tree/base/model
    # save the result in a variable. 
    # Strange syntax, because we want to save *all* entries from `result` in `model_names`.
    model_names=( ${result[@]} )
    # print the first element from that `result` array we just got
    printf 'First element is %s. Total elements %d.\n' "$model_names" "${#model_names[@]}"
    printf 'All elements concatenated with "foobar" in between:'
    printf '%sfoobar' ${model_names[@]}
    
    function simple_devicetree_string() {
      local tmp
      mapfile -d '' tmp < $1
      printf '%s ' ${tmp[@]}
    }
    
    printf 'Let's read this simply into a line, safely: %s\n' "$(simple_devicetree_string /proc/device-tree/base/model)"
    

    除此之外,您不应该/sys/firmware/devicetree/base直接使用这些路径,而应该将其用作/proc/device-tree/base/前缀。这保证在下一次内核更新时仍然有效,但可能会消失。(这些实际上/sys/firmware/devicetree/不是存储在某个地方的文件——这只是 sysfs 中表示的解析的设备树结构。它是在您读取时由内核计算的,而不是存储在磁盘上的某个文件。因此,它们的存在基本上是这里内核的 API——内核开发人员说“请通过 /proc 访问,另一部分是我们当前使用的,但我们不能保证它不会改变。/proc 路径,我们保证。”。)

    • 1
  3. Best Answer
    G-Man Says 'Reinstate Monica'
    2024-07-07T07:26:12+08:002024-07-07T07:26:12+08:00

    按照我对抑制 echo 命令的执行跟踪?的回答,尝试

    { rpi_model_name=$(cat /proc/device-tree/model); }            2>/dev/null
    { rpi_serial_number=$(cat /proc/device-tree/serial-number); } 2>/dev/null
    

    或者

    {
      rpi_model_name=$(cat /proc/device-tree/model)
      rpi_serial_number=$(cat /proc/device-tree/serial-number)
    } 2>/dev/null
    
    • 1

相关问题

  • 在awk中的两行之间减去相同的列

  • 打印文件行及其长度的脚本[关闭]

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 按分隔符拆分并连接字符串问题

  • MySQL Select with function IN () with bash array

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve