Sie verwechseln zwei sehr unterschiedliche Arten von Eingaben: STDIN und Argumente. Argumente sind eine Liste von Zeichenfolgen, die dem Befehl beim Start bereitgestellt werden, in der Regel durch Angabe nach dem Befehlsnamen (z . B. echo these are some arguments
oder rm file1 file2
). STDIN hingegen ist ein Strom von Bytes (manchmal Text, manchmal nicht), den der Befehl (optional) nach dem Start lesen kann. Hier sind einige Beispiele (Anmerkung, cat
die entweder Argumente oder STDIN annehmen kann, aber verschiedene Dinge damit macht):
echo file1 file2 | cat # Prints "file1 file2", since that's the stream of
# bytes that echo passed to cat's STDIN
cat file1 file2 # Prints the CONTENTS of file1 and file2
echo file1 file2 | rm # Prints an error message, since rm expects arguments
# and doesn't read from STDIN
xargs
Man kann sich vorstellen, wie man eine Eingabe im STDIN-Stil in Argumente umwandelt:
echo file1 file2 | cat # Prints "file1 file2"
echo file1 file2 | xargs cat # Prints the CONTENTS of file1 and file2
echo
Tatsächlich macht es mehr oder weniger das Gegenteil: Es konvertiert seine Argumente in STDOUT (das an die STDIN eines anderen Befehls weitergeleitet werden kann):
echo file1 file2 | echo # Prints a blank line, since echo doesn't read from STDIN
echo file1 file2 | xargs echo # Prints "file1 file2" -- the first echo turns
# them from arguments into STDOUT, xargs turns
# them back into arguments, and the second echo
# turns them back into STDOUT
echo file1 file2 | xargs echo | xargs echo | xargs echo | xargs echo # Similar,
# except that it converts back and forth between
# args and STDOUT several times before finally
# printing "file1 file2" to STDOUT.
ls | grep -v "notes.txt" | xargs rm
, um alles zu entfernen, es sei dennnotes.txt
, Sie analysieren niemals diels
Ausgabe . Ihr Befehl würde abbrechen, wenn eine einzelne Datei beispielsweise ein Leerzeichen enthält. Der sicherere Weg wärerm !(notes.txt)
in Bash (mitshopt -s extglob
Satz) oderrm ^notes.txt
in Zsh (mitEXTENDED_GLOB
) usw.