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 / computer / Perguntas / 1470243
Accepted
Hashbrown
Hashbrown
Asked: 2019-08-12 20:35:09 +0800 CST2019-08-12 20:35:09 +0800 CST 2019-08-12 20:35:09 +0800 CST

Por que a tubulação não funciona para um comando que espera stdin?

  • 772

Então, eu estava procurando colocar a saída unzip -Z1em uma matriz e encontrei esta resposta ; sua primeira opção, usando mapfilee processando substituição COM redirecionamento de entrada, funciona muito bem.

Mas então pensei: "espere, substituição de processo", que cria um descritor de arquivo a partir do stdout, "então use o redirecionamento de entrada para colocar o conteúdo no stdin?"
Isso não é equivalente a apenas tubulação?

Aparentemente não. Não, não é. Aqui tento colocar o conteúdo de lsdentro de uma variável, mas usando um pipe: à la ls | mapfile -t test
insira a descrição da imagem aqui Nada.

Mas, se eu seguir essa resposta para um T: id est mapfile -t test < <(ls)
insira a descrição da imagem aqui Voilà.

Mas por quê? diffnão posso dizer a diferença. Pelo menos entre seu conteúdo .
insira a descrição da imagem aqui

Não consegui encontrar nada inerentemente especial sobre < <(), além de criar um descritor de arquivo. Usando isso como teoria, tentei um HEREDOC, que faz um descritor de arquivo também. Funcionou:

mapfile -t test2 << END && printf '%s ' "${test2[@]}"
this
is
an
array
END

Mas não se for lavado stdinvia cat, tente:

cat << END | mapfile -t test3 && printf '%s ' "${test3[@]}"
this
is
an
array
END

Então, minhas perguntas.
Essa teoria do descritor de arquivo/descoberta do HEREDOC não deveria importar mapfile, pois todos os seus manuais dizem que ele usa "entrada padrão", certo?
Se 'entrada padrão' de alguma forma significa que deve ser um descritor mais sofisticado; porque? Por que ele não consegue usar um fluxo de entrada simples de leitura única para gerar o array?
E, finalmente, se ambos forem respondidos/no caminho certo, por que o comando não falhou quando esperava um fd? Você verá que meu terminal 0imprimiu um monte (e usamos &&para os últimos exemplos), isso está $?no meu prompt, então não houve nenhum erro relatado pelo builtin.

Eu tentei isso no Fedora 30 e RHEL 6, então não acho que seja um bug.

bash stdin
  • 1 1 respostas
  • 836 Views

1 respostas

  • Voted
  1. Best Answer
    user1686
    2019-08-12T21:07:19+08:002019-08-12T21:07:19+08:00

    Funciona; é que os resultados desaparecem imediatamente depois.

    A diferença é que foo | mapfileé um pipeline que consiste em dois novos processos – cada elemento, seja um programa externo ou um shell integrado, é executado em um subshell. Portanto, embora o mapfile ainda faça seu trabalho, ele não tem como transferir o resultado para o processo do shell pai.

    Para comparação, você obteria o mesmo resultado com (mapfile < foo.txt)ou ls | while read line; do ... doneou mesmo (foo=bar); echo $foo. Em todos esses casos, se você comparar $BASHPID e $$, verá valores diferentes – o primeiro mostra o PID do processo do subshell sendo diferente do PID do shell principal.

    Ao usar um redirecionamento, o mapfilecomando é executado no processo principal do shell e pode alterar as variáveis ​​do shell corretamente. Não importa muito se está sendo redirecionado de um arquivo real, ou de um <<arquivo temporário gerado, ou de um <()arquivo mágico gerado - neste caso, o shell lida com isso como uma etapa separada e não causa um pipeline ser construido. (O comando interno de <()ou $() é um subshell, é claro.)


    Observe que a variável pode realmente ser usada no subshell – ela simplesmente não pode ser transferida para cima. Portanto, para operações únicas, a 2ª parte do pipeline pode ser bastante elaborada:

    unzip -l | {
        mapfile -t test
        for file in "${test[@]}"; do
            echo "I got $file!"
        done
    }
    
    zipinfo | while read -r file; do
        echo "I got $file!"
    done
    
    • 2

relate perguntas

  • substituindo zsh por bash no usuário não root

  • Tendo problemas para definir variáveis ​​de ambiente no Terminal no macOS High Sierra

  • Existe um equivalente a cd - para cp ou mv?

  • Notificar-enviar notificações aparecendo na janela

  • como abrir um arquivo de escritório do WSL

Sidebar

Stats

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

    O visualizador de fotos do Windows não pode ser executado porque não há memória suficiente?

    • 5 respostas
  • Marko Smith

    Como faço para ativar o WindowsXP agora que o suporte acabou?

    • 6 respostas
  • Marko Smith

    Área de trabalho remota congelando intermitentemente

    • 7 respostas
  • Marko Smith

    Serviço do Windows 10 chamado AarSvc_70f961. O que é e como posso desativá-lo?

    • 2 respostas
  • Marko Smith

    O que significa ter uma máscara de sub-rede /32?

    • 6 respostas
  • Marko Smith

    Ponteiro do mouse movendo-se nas teclas de seta pressionadas no Windows?

    • 1 respostas
  • Marko Smith

    O VirtualBox falha ao iniciar com VERR_NEM_VM_CREATE_FAILED

    • 8 respostas
  • Marko Smith

    Os aplicativos não aparecem nas configurações de privacidade da câmera e do microfone no MacBook

    • 5 respostas
  • Marko Smith

    ssl.SSLCertVerificationError: falha na verificação do certificado [SSL: CERTIFICATE_VERIFY_FAILED]: não foi possível obter o certificado do emissor local (_ssl.c:1056)

    • 4 respostas
  • Marko Smith

    Como posso saber em qual unidade o Windows está instalado?

    • 6 respostas
  • Martin Hope
    Albin Como faço para ativar o WindowsXP agora que o suporte acabou? 2019-11-18 03:50:17 +0800 CST
  • Martin Hope
    fixer1234 O "HTTPS Everywhere" ainda é relevante? 2019-10-27 18:06:25 +0800 CST
  • Martin Hope
    Kagaratsch O Windows 10 exclui muitos arquivos minúsculos muito lentamente. Algo pode ser feito para agilizar? 2019-09-23 06:05:43 +0800 CST
  • Martin Hope
    andre_ss6 Área de trabalho remota congelando intermitentemente 2019-09-11 12:56:40 +0800 CST
  • Martin Hope
    Riley Carney Por que colocar um ponto após o URL remove as informações de login? 2019-08-06 10:59:24 +0800 CST
  • Martin Hope
    zdimension Ponteiro do mouse movendo-se nas teclas de seta pressionadas no Windows? 2019-08-04 06:39:57 +0800 CST
  • Martin Hope
    Inter Sys Como Ctrl+C e Ctrl+V funcionam? 2019-05-15 02:51:21 +0800 CST
  • Martin Hope
    jonsca Todos os meus complementos do Firefox foram desativados repentinamente, como posso reativá-los? 2019-05-04 17:58:52 +0800 CST
  • Martin Hope
    MCK É possível criar um código QR usando texto? 2019-04-02 06:32:14 +0800 CST
  • Martin Hope
    SoniEx2 Altere o nome da ramificação padrão do git init 2019-04-01 06:16:56 +0800 CST

Hot tag

windows-10 linux windows microsoft-excel networking ubuntu worksheet-function bash command-line hard-drive

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