AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / unix / Perguntas / 458760
Accepted
Harry
Harry
Asked: 2018-07-27 22:57:12 +0800 CST2018-07-27 22:57:12 +0800 CST 2018-07-27 22:57:12 +0800 CST

PID da função em segundo plano, F, em comandos invocados em subshells dentro de F

  • 772

Quando fooexecutado em segundo plano, o BASHPIDof foo( bashpid_of_foo) não está disponível dentro dos corpos bar_1para bar_nvia $BASHPID, pois eles são invocados por meio do recurso de substituição de comando de bash:

function foo() {
    local bashpid_of_foo=$BASHPID
    local output

    # desired to be shared by all Command Substitutions
    # in the body of this function.
    local log=/path/to/log.$BASHPID

    ... >> $log

    output=$(bar_1 ...)
      ...
    output=$(bar_n ...)
}

function bar_1() {
   # log only specific (and NOT all) messages
   # to the shared log file of the invoking thread.
    ... >> /path/to/log.$BASHPID
}

foo &
foo &

Pergunta: Existe uma maneira elegante de contornar a limitação acima, sem ter que passar bashpid_of_foopor variáveis ​​de ambiente adhoc ou arquivos de disco externo?

Por elegante , quero dizer, ser capaz de manter as interfaces e os corpos das bar_*funções limpos confiando apenas nos recursos fornecidos pelo bash. (Por exemplo BASHPID, é um bashrecurso.)

Se eu tentar substituir o valor de BASHPID, assim,

out_1=$(BASHPID=$BASHPID bar_1 ...)

... ele (corretamente) reclama de BASHPIDser uma variável readonly.

EDIT: (1) Adicionada a definição bar_1acima. (2) Adicionado a 2ª chamada fooem segundo plano. Cada foochamada precisa manter seu próprio arquivo de log, pois gravar em um arquivo comum pode resultar em conteúdo ilegível.

NOTA: Qualquer que seja o log que aconteça no contexto de tempo de execução do foo, quero que ele vá para o fooarquivo de log específico, /path/to/log.$BASHPID SEM passar o nome desse arquivo de log ou mesmo o PID do foo. foopode ter várias instâncias em execução em segundo plano.

bash process
  • 2 2 respostas
  • 64 Views

2 respostas

  • Voted
  1. slm
    2018-07-27T23:31:03+08:002018-07-27T23:31:03+08:00

    OBSERVAÇÃO: em relação às suas perguntas gerais sobre $BASHPID, isso é somente leitura e você não pode manipulá-lo. Isso é por design.

    Normalmente, meu conselho geral é que, quando você chegar ao nível de sofisticação com scripts Bash como esse, é hora de movê-lo para Python, Ruby, o que for.


    Seu exemplo

    Acho que não vejo qual é o seu problema, isso funciona para mim:

    $ cat subby.bash
    #!/bin/bash
    
    function foo() {
        local bashpid_of_foo=$BASHPID
        local output=blipblop
    
        echo "foo: $bashpid_of_foo"
        echo "foo: $output"
        out_1=$(echo $bashpid_of_foo)
        out_n=$(echo $output)
        echo "out_1: $out_1"
        echo "out_n: $out_n"
    }
    
    foo &
    
    
    $ ./subby.bash
    foo: 4900
    foo: blipblop
    out_1: 4900
    out_n: blipblop
    

    Adição de exportação

    Se mudarmos isso transformando bar_1em um script de shell:

    $ cat bar_1
    #!/bin/bash
    
    echo "from bar_1: $bashpid_of_foo"
    

    E altere seu script original assim:

    #!/bin/bash
    
    function foo() {
        export bashpid_of_foo=$BASHPID
        local output=blipblop
    
        echo "foo: $bashpid_of_foo"
        echo "foo: $output"
        out_1=$(echo $bashpid_of_foo)
        out_2=$(./bar_1)
        out_n=$(echo $output)
        echo "out_1: $out_1"
        echo "out_2: $out_2"
        echo "out_n: $out_n"
    }
    
    foo &
    

    Podemos ver que o $bashpid_of_fooestá sendo exportado corretamente para subshells:

    $ ./subby.bash
    foo: 5014
    foo: blipblop
    out_1: 5014
    out_2: from bar_1: 5014
    out_n: blipblop
    

    Precisamos usar um exportaqui e não apenas um localporque, caso contrário, as variáveis ​​de ambiente não serão exportadas para nenhum filho. Os subshells são shells filhos aqui.

    $ help export
    ...
       Marks each NAME for automatic export to the environment of subsequently
        executed commands.  If VALUE is supplied, assign VALUE before exporting.
    
    • 1
  2. Best Answer
    Kusalananda
    2018-07-28T01:28:57+08:002018-07-28T01:28:57+08:00
    #!/bin/bash
    
    bar () {
        # bashpid is set in our environment from the calling function
        printf 'bar BASHPID = %d, bar bashpid = %d\n' "$BASHPID" "$bashpid"
    
        # in your case, you would have...
    
        local logfile="/some/path/to/log.$bashpid"
    
        # etc.
    }
    
    foo () {
        local bashpid="$BASHPID"
        local message
    
        local logfile="/some/path/to/log.$BASHPID"
    
        message=$( bashpid="$bashpid" bar ); printf 'Message from bar: %s\n' "$message"
        message=$( bashpid="$bashpid" bar ); printf 'Message from bar: %s\n' "$message"
        message=$( bashpid="$bashpid" bar ); printf 'Message from bar: %s\n' "$message"
        message=$( bashpid="$bashpid" bar ); printf 'Message from bar: %s\n' "$message"
    }
    
    foo &
    foo &
    foo &
    
    wait
    

    Exemplo de execução:

    $ bash script.sh
    Message from bar: bar BASHPID = 71979, bar bashpid = 18461
    Message from bar: bar BASHPID = 7420, bar bashpid = 71036
    Message from bar: bar BASHPID = 6109, bar bashpid = 18461
    Message from bar: bar BASHPID = 27868, bar bashpid = 71036
    Message from bar: bar BASHPID = 44547, bar bashpid = 60086
    Message from bar: bar BASHPID = 69310, bar bashpid = 71036
    Message from bar: bar BASHPID = 37649, bar bashpid = 60086
    Message from bar: bar BASHPID = 15999, bar bashpid = 71036
    Message from bar: bar BASHPID = 81520, bar bashpid = 18461
    Message from bar: bar BASHPID = 92568, bar bashpid = 60086
    Message from bar: bar BASHPID = 72438, bar bashpid = 18461
    Message from bar: bar BASHPID = 15094, bar bashpid = 60086
    

    Para cada chamada de foona parte principal do script, quatro chamadas para barserão feitas e quatro linhas de saída serão produzidas. Como você vê, existem apenas três bashpidnúmeros únicos, cada um de uma das fooinvocações.

    Outra maneira de passar $bashpidde footo baré obviamente passá-lo como um argumento de linha de comando e recebê-lo com local bashpid="$1"in barou algo semelhante, mas você diz que não quer fazer isso.

    • 1

relate perguntas

  • exportar variáveis ​​​​env programaticamente, via stdout do comando [duplicado]

  • Problema estranho ao passar variáveis ​​do arquivo de texto

  • Enquanto a linha lê mantendo os espaços de escape?

  • ordem de substituição de processos `te` e `bash`

  • Execute um script muito lento até que seja bem-sucedido

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Como exportar uma chave privada GPG e uma chave pública para um arquivo

    • 4 respostas
  • Marko Smith

    ssh Não é possível negociar: "nenhuma cifra correspondente encontrada", está rejeitando o cbc

    • 4 respostas
  • Marko Smith

    Como podemos executar um comando armazenado em uma variável?

    • 5 respostas
  • Marko Smith

    Como configurar o systemd-resolved e o systemd-networkd para usar o servidor DNS local para resolver domínios locais e o servidor DNS remoto para domínios remotos?

    • 3 respostas
  • Marko Smith

    Como descarregar o módulo do kernel 'nvidia-drm'?

    • 13 respostas
  • Marko Smith

    apt-get update error no Kali Linux após a atualização do dist [duplicado]

    • 2 respostas
  • Marko Smith

    Como ver as últimas linhas x do log de serviço systemctl

    • 5 respostas
  • Marko Smith

    Nano - pule para o final do arquivo

    • 8 respostas
  • Marko Smith

    erro grub: você precisa carregar o kernel primeiro

    • 4 respostas
  • Marko Smith

    Como baixar o pacote não instalá-lo com o comando apt-get?

    • 7 respostas
  • Martin Hope
    rocky Como exportar uma chave privada GPG e uma chave pública para um arquivo 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add retorna com: "Erro ao conectar ao agente: nenhum arquivo ou diretório" 2018-08-24 23:28:13 +0800 CST
  • Martin Hope
    Evan Carroll status systemctl mostra: "Estado: degradado" 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim Como podemos executar um comando armazenado em uma variável? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S Por que /dev/null é um arquivo? Por que sua função não é implementada como um programa simples? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 Como ver as últimas linhas x do log de serviço systemctl 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - pule para o final do arquivo 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla Por que verdadeiro e falso são tão grandes? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis Substitua a string em um arquivo de texto enorme (70 GB), uma linha 2017-12-30 06:58:33 +0800 CST
  • Martin Hope
    Bagas Sanjaya Por que o Linux usa LF como caractere de nova linha? 2017-12-20 05:48:21 +0800 CST

Hot tag

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

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve