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 / 问题 / 691539
Accepted
Yzmir Ramirez
Yzmir Ramirez
Asked: 2022-02-21 23:25:02 +0800 CST2022-02-21 23:25:02 +0800 CST 2022-02-21 23:25:02 +0800 CST

如何在 bash 中倒带文件描述符而不使用第二个文件描述符从 bash for Mac 的开头读取?

  • 772

环境: GNU bash,版本 3.2.57(1)-release (x86_64-apple-darwin20)

尝试0:

exec 3<> "$(mktemp)"    # open file descriptor 3 to a temp file for read/write
echo 'foo' >&3          # write 'foo' to descriptor 3
read bar <&3            # read contents of descriptor 3 to variable named bar (not using -u 3)
echo $?                 # exit status returns "1"
exec 3>&-               # close file descriptor 3

在上面的代码片段中,我希望$bar设置为"foo"但它不起作用,因为文件描述符 3 中的位置位于文件描述符文件的末尾。

尝试1:

exec 3<> "$(mktemp)"    # open file descriptor 3 to a temp file for read/write
echo 'foo' >&3          # write 'foo' to descriptor 3
cat <&3                 # echo contents of of descriptor 3 to STDOUT
echo $?                 # exit status returns "0"
exec 3>&-               # close file descriptor 3

这也失败了,因为描述符 3 中的文件位置位于末尾。

在 C/C++ 中,您可以rewind(...) 和 fseek(...)将创建的文件临时文件的文件指针重置到文件mktemp的开头。

此处已知的解决方法:(使用第二个描述符)

myTempFile="$(mktemp)"  # create temp file and assign to $myTempFile
exec 3> "$myTempFile"   # open file descriptor 3 for writing to $myTempFile
exec 4< "$myTempFile"   # open file descriptor 4 for reading of $myTempFile
echo 'foo' >&3          # write 'foo' to descriptor 3
read bar <&4            # read contents of descriptor 4 to variable named bar
echo $?                 # exit status returns "0"
echo "$bar"             # echo "foo"
exec 3>&-               # close file descriptor 3
exec 4>&-               # close file descriptor 4

问题™(最终):

如何在 bash 中回退文件描述符而不使用第二个文件描述符从头开始读取?

注意:我不想安装ksh93来使用它的倒带功能。

exec 3<> "$(mktemp)"
echo 'foo' >&3
# Magic Happens here
read bar <&3             # expect $bar == "foo"
echo $?                  # expect "0"
exec 3>&- 
bash darwin
  • 3 3 个回答
  • 294 Views

3 个回答

  • Voted
  1. ctrl-alt-delor
    2022-02-22T01:01:48+08:002022-02-22T01:01:48+08:00

    虽然这是一个有趣的问题(但也是我 30 年来从未需要用外壳做的事情)。您可能正在尝试使用错误的工具。

    Unix(UNIX、BSD、MacOS、Gnu/Linux)有管道。管道是一个特殊的文件(不在二级存储/磁盘中),它有两个文件描述符。一种用于写作,一种用于阅读。读描述符总是跟在写之后。读完数据后就废掉了。

    echo a_command | another_command
    

    有时您无法在启动命令时创建管道。在这些情况下,您可以使用命名管道或 unix-sockets。

    使用其中一个进程创建和销毁一个 unix-socket。这也是两种方式。

    可以独立于两个进程创建命名管道。

    • 4
  2. Stéphane Chazelas
    2022-02-21T23:58:32+08:002022-02-21T23:58:32+08:00

    zsh并且ksh93具有系统调用的内置包装器lseek(),但没有 bash,因此您需要调用一个单独的实用程序,该实用程序可以对附加到 fdlseek()的打开文件描述执行 zsh默认情况下安装在 macos 上,因此,如果您必须使用bash,您可以定义一个包装的内置sysseek函数:zshsysseek

    sysseek() {
      zsh -c '
        zmodload zsh/system
        sysseek "$@" || syserror -p "$0: "
      ' sysseek "$@"
    }
    

    并使用:

    sysseek -u 3 0
    

    将 fd 3 倒回到开头。

    没有系统的另一种选择zsh是使用perl更广泛可用的系统:

    sysseek() { # args: fd, mode, offset
      perl -e '
        my ($fd, $mode, $offset) = @ARGV;
        open FD, ">&", $fd or die "fd $fd: $!\n";
        seek FD, $mode, $offset or die "seek: $!\n";
      ' -- "$@"
    }
    

    并使用:

    sysseek 3 0 0
    

    将 fd 3 倒回到开头。

    第二个参数可以是:

    1. seek 相对于开始(如sysseek -w startin zsh,默认存在)。
    2. 相对于当前位置 ( zsh's sysseek -w current)寻找
    3. 相对于文件末尾 ( zsh's sysseek -w end) 查找。
    • 2
  3. Best Answer
    nezabudka
    2022-02-22T05:49:13+08:002022-02-22T05:49:13+08:00

    将读取描述符的创建与写入描述符分开:

    exec 3>file
    echo foo >&3
    exec 3<file
    cat <&3
    exec 3<&-
    

    3 描述符的表被覆盖,变为只读。使用通用描述符是没有意义的。

    在变量中隐藏描述符编号:

    { echo foo >&$fd; exec {fd}<&-; } {fd}<>file
    { cat <&$fd; } {fd}<>file
    exec {fd}<&-
    

    由于首先执行重定向,因此$fd变量被写入并且在块内已经有一个值。这里需要注意的是,句柄不会像 ksh93 中那样自动关闭,必须显式关闭,在块的末尾或使用下一个命令(显示两个选项)。准备在下一个 bash 版本中引入一个内置变量,可以将其更改为自动关闭行为。使用这种形式的调用,不需要命令exec {fd}<&-

    • 1

相关问题

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

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

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