Версия для печати
Нажмите сюда для просмотра этой темы в оригинальном формате
Форум на Исходниках.RU > *nix > пернаправление ввода


Автор: v4567 29.07.17, 11:34
Есть скрипт на BASH, в котором я перехватываю сигналы:
<{CODE_COLLAPSE_OFF}><{CODE_WRAP_OFF}>
    # Перехватываем сигналы.
     # Завершение работы, сигнал "TERM".
     trap 'vozvrisxzn ; exit 0' SIGTERM
     # Завершение работы, сигнал посланный комбинацией клавиш Ctrl+c.
     trap 'vozvrisxzn ; exit 0' SIGINT
     # Отправка в фон - нажатие Ctrl+z.
     trap 'vozvrisxzn' SIGTSTP
     # Возврат из фона - набор в консоли bg или fg.
     trap 'izfona' SIGCONT
     # Изменение размеров терминала во время работы программы.
     trap 'izmrazmterm' SIGWINCH


Этот код стоит в начале скрипта. Далее после некоторого выполнения скрипта (он выводит некоторую информацию в терминал) он останавливается и ждёт нажатия клавиш командой read. Если вместо read сделать бесконечный пустой цикл, то сигналы обрабатываются сразу. Например после изменения размеров графического эмулятора терминала, выведенная информация отлично подстраивается под изменившиеся размеры терминала.
Когда же мы ждём нажатия клавиш командой read, то после изменения размеров терминала обработка этого сигнала происходит только тогда когда мы нажмём какую нибудь клавишу, то есть только тогда когда отработает команда read. Получается, что обработка сигналов происходит только после окончания выполнения команды во время которой пришёл сигнал.
Поставить маленькое время ожидания read (например -t 0.5 секунд) я не могу, мне же надо что бы сигналы обрабатывались сразу или хотя бы с маленькой задержкой, но не после тога как я нажму на клавишу.
Вариант вижу только один (да и то я не уверен что это можно сделать), read сделать в отдельной функции например fynk() и эту функцию запускать в фоне. Получится так что запуститься дополнительный экземпляр bash с этой функцией, его pid я могу узнать, получаем вот такой код:
<{CODE_COLLAPSE_OFF}><{CODE_WRAP_OFF}>
    fynk &
    PIDFYNK=$!


Теперь вопрос и трудность вот в чём, можно ли как то перенаправить ввод с клавиатуры то есть перенаправить стандартный входной поток (дескриптор файла 0) в этот процесс, функцию fynk и команду в этой функции read? Если можно то как это сделать?

Получается так что у меня на терминал выводит один процесс, а клавиатура должна перенаправляться в процесс работающий в фоне.

Думаю если так сделать можно, то и сигналы будут обрабатываться быстро, так как у меня в процессе запущенном непосредственно в терминале не будет команды read.

За помощь буду заранее благодарен!

Автор: v4567 02.08.17, 16:27
Если использовать просто read то работает.

Но у меня вот такая конструкция:

<{CODE_COLLAPSE_OFF}><{CODE_WRAP_OFF}>
    while [ true ]
      do
     
        KEY=""
        zaderjka=""
     
        for (( i=0 ; i<10 ; i++ ))
         do
           eval $( (IFS="" read -r -s -n 1 $zaderjka key ; printf 'code %d' "'$key") 2>&1 | awk 'NR==1||NR==4 {print $1 "=" $2}' )
           if [ "$code" != "0" ] || [ "$code" = "0" -a "$zaderjka" = "" ]
            then
                KEY=$KEY$code
           fi
           zaderjka="-t 0.01"
        done
    done


Вот с такой конструкцией всё останавливается и ждёт нажатия клавиши.

На другом форуме подсказали вот такую конструкцию:

<{CODE_COLLAPSE_OFF}><{CODE_WRAP_OFF}>
    while [ true ]
      do
     
        KEY=""
        zaderjka=""
     
        for (( i=0 ; i<10 ; i++))
         do
           code=""
           IFS="" read -r -s -n 1 $zaderjka key && printf -v code '%d' "'$key"
           [ -n "$code" ] && KEY=$KEY$code
           zaderjka="-t 0.01"
        done
    done


При такой конструкции всё нормально, почему при первой останавливалось я так и не понял.

Powered by Invision Power Board (https://www.invisionboard.com)
© Invision Power Services (https://www.invisionpower.com)