Как использовать переменную для обозначения дескриптора файла в bash? - PullRequest
22 голосов
/ 28 ноября 2011

Я хочу использовать переменную bash для обозначения дескриптора файла, например:

id=6
file=a
exec $id<>$file

Но использование неверно:

-bash: exec: 6: not found

Итак, как использовать переменнуюуказать дескриптор файла в команде exec?

Ответы [ 3 ]

18 голосов
/ 21 сентября 2015

Принятый ответ правильный, но начиная с bash 4.1 , вы можете использовать автоматическое распределение файловых дескрипторов, и в этом случае вам не нужно eval:

file=a
exec {id}<>"$file"

Тогда вы можете использовать это так:

echo  test >&${id}

или

fsck -v -f -C ${id} /dev/something
14 голосов
/ 28 ноября 2011

Вы должны использовать eval и поместить все выражение в кавычки.

eval "exec $id<>$file"

И делайте это каждый раз, когда хотите использовать $id.

0 голосов
/ 02 июля 2018

Я нашел обсуждение в ответе tobias.pal очень интересным: https://stackoverflow.com/a/32689974/1184842

for (( FD=3 ; FD < 100 ; FD++ )) ; do exec {FD}> file.$FD ; echo $FD >&${FD}; done

Это не сработает, поскольку exec {FD}> file.${FD} будет одним и тем же дескриптором для всех значений $ FD, верно? (Денио Мариз)

Я решил это с помощью массива, заявленного Дрю Чапином:

#!/bin/bash
# global variables for temp file handling
declare -a TEMPORARY_FILES_WRITE;
declare -a TEMPORARY_FILES_READ;

function createTempFile() {
    local usecase="$1"
    local id="$2"
    local tmpfile=$(mktemp)  # Create a temporal file in the default temporal folder of the system

    # Lets do some magic for the tmpfile to be removed when this script ends, even if it crashes
    exec {TEMPORARY_FILES_WRITE[$id]}>"$tmpfile"
    exec {TEMPORARY_FILES_READ[$id]}<"$tmpfile"
    rm "$tmpfile"  # Delete the file, but file descriptors keep available for this script
}    

for (( FD=3 ; FD < 100 ; FD++ )) ; do 
    TEMP_FILE_COUNTER=$((TEMP_FILE_COUNTER + 1))
    createTempFile "Iteration $FD" $FD ;
    echo $FD >&${TEMPORARY_FILES_WRITE[$FD] ;
done

example=$(cat <&${TEMPORARY_FILES_READ[50]})
echo $example

Это выдаст 50.

...