Heim > Artikel > Backend-Entwicklung > Befehlszeilentools mit Go: Daten weiterleiten
Unix ist dafür bekannt, die Philosophie zu vertreten, dass Befehle eine Sache tun sollten, und zwar gut.
Ausgefeilte Datenverarbeitungs- und Transformationsoperationen können oft mithilfe des Shell-Pipe-Operators durchgeführt werden, um Befehle miteinander zu verketten, sodass die Ausgabe eines Befehls zur Eingabe eines anderen wird, wodurch Daten manipuliert und transformiert werden, um das gewünschte Ergebnis zu erzielen.
Zum Beispiel:
# Sort file names. ls | sort # Count files. ls -l | count -l # Print out unique file extensions. # 1. List all files that have extensions # 2. Transform the data (discard everything but extensions) # 3. Sort the list (data must be sorted to identify duplicates) # 4. Filter out duplicates # 5. Browse the results ls *.* | sed 's/.*\.//' | sort | uniq | less
Mit Go können Programmierer effiziente und leistungsstarke Befehle zur Datenverarbeitung erstellen. Wir werden darauf mit den folgenden Auszügen eingehen.
Das Wesentliche an einem Befehl, der in einer Pipe-Operation verwendet werden kann, besteht darin, dass er von stdin liest und in stdout schreibt.
add-line-numbers.go
package main import ( "bufio" "fmt" "os" ) func main() { // Buffered input that splits input on lines. input := bufio.NewScanner(os.Stdin) // Buffered output. output := bufio.NewWriter(os.Stdout) lineNo := 0 // Scan until EOF (no more input). for input.Scan() { text := input.Text() lineNo++ s := fmt.Sprintf("%03d %s\n", lineNo, text) // It would be simpler to just use fmt.Println, // but want to emphasize piping stdin to stdout // explicitly. // Intentionally ignoring return values. _, _ = output.WriteString(s) } // Always explicitly flush remaining buffered output. _ = output.Flush() }
Dieses Beispiel liest jeweils eine Zeile von stdin und schreibt sie zurück nach stdout, wobei jeder Zeile die Zeilennummer vorangestellt wird. Hier verwenden wir die Programmdatei selbst als Eingabe, um eine nummerierte Ausgabe zu generieren.
$ cat add-line-numbers.go | go run add-line-numbers.go 001 package main 002 003 import ( 004 "bufio" 005 "fmt" 006 "os" 007 ) 008 009 func main() { 010 011 // Buffered input that splits input on lines. 012 input := bufio.NewScanner(os.Stdin) 013 014 // Buffered output. 015 output := bufio.NewWriter(os.Stdout) 016 017 lineNo := 0 018 019 // Scan until EOF (no more input). 020 for input.Scan() { 021 text := input.Text() 022 lineNo++ 023 s := fmt.Sprintf("%03d %s\n", lineNo, text) 024 025 // It would be simpler to just use fmt.Println, 026 // but want to emphasize piping stdin to stdout 027 // explicitly. 028 // Intentionally ignoring return values. 029 _, _ = output.WriteString(s) 030 031 } 032 033 // Always explicitly flush remaining buffered output. 034 _ = output.Flush() 035 036 }
Dieses Beispiel liest jeweils eine Zeile aus stdin, kodiert sie mit Base64 und schreibt sie zurück nach stdout.
package main import ( "bufio" "encoding/base64" "os" ) func main() { // Buffered input that splits input on lines. input := bufio.NewScanner(os.Stdin) // Base64 Encoder/writer. encoder := base64.NewEncoder( base64.StdEncoding, os.Stdout) // Scan until EOF (no more input). for input.Scan() { bytes := input.Bytes() _, _ = encoder.Write(bytes) _, _ = encoder.Write([]byte{'\n'}) } // Close the encoder and ensure it flushes remaining output _ = encoder.Close() }
Da der Scanner Zeilenumbruchzeichen (n) aufteilt, ohne sie zurückzugeben, ist es notwendig, nach dem Schreiben jeder Zeile explizit einen Zeilenumbruch zu schreiben.
$ cat base64-encode.go | go run base64-encode.go cGFja2FnZSBtYWluCgppbXBvcnQgKAoJImJ1ZmlvIgoJImVuY29kaW5nL2Jhc2U2NCIKCSJvcyIKKQoKZnVuYyBtYWluKCkgewoKCS8vIEJ1ZmZlcmVkIGlucHV0IHRoYXQgc3BsaXRzIGlucHV0IG9uIGxpbmVzLgoJaW5wdXQgOj0gYnVmaW8uTmV3U2Nhbm5lcihvcy5TdGRpbikKCgkvLyBCYXNlNjQgRW5jb2Rlci93cml0ZXIuCgllbmNvZGVyIDo9IGJhc2U2NC5OZXdFbmNvZGVyKAoJCWJhc2U2NC5TdGRFbmNvZGluZywKCQlvcy5TdGRvdXQpCgoJLy8gU2NhbiB1bnRpbCBFT0YgKG5vIG1vcmUgaW5wdXQpLgoJZm9yIGlucHV0LlNjYW4oKSB7CgkJYnl0ZXMgOj0gaW5wdXQuQnl0ZXMoKQoJCV8sIF8gPSBlbmNvZGVyLldyaXRlKGJ5dGVzKQoJCV8sIF8gPSBlbmNvZGVyLldyaXRlKFtdYnl0ZXsnXG4nfSkKCX0KCgkvLyBDbG9zZSB0aGUgZW5jb2RlciBhbmQgZW5zdXJlIGl0IGZsdXNoZXMgcmVtYWluaW5nIG91dHB1dAoJXyA9IGVuY29kZXIuQ2xvc2UoKQp9Cg==
Sie können bestätigen, dass der Text korrekt codiert wurde, indem Sie das codierte Ergebnis an den Base64-Systembefehl (Linux und MacOS) weiterleiten, um ihn zu decodieren:
$ cat base64-encode.go | go run base64-encode.go | base64 -D package main import ( "bufio" "encoding/base64" "os" ) func main() { // Buffered input that splits input on lines. input := bufio.NewScanner(os.Stdin) // Base64 Encoder/writer. encoder := base64.NewEncoder( base64.StdEncoding, os.Stdout) // Scan until EOF (no more input). for input.Scan() { bytes := input.Bytes() _, _ = encoder.Write(bytes) _, _ = encoder.Write([]byte{'\n'}) } // Close the encoder and ensure it flushes remaining output _ = encoder.Close() }
Dieser Beitrag ist ein Auszug aus einem kurzen Einführungsleitfaden, den ich über Standardbibliotheksfunktionen von Go geschrieben habe, die für die Erstellung von Befehlszeilentools nützlich sind: Go für CLI-Apps und -Tools.
Das obige ist der detaillierte Inhalt vonBefehlszeilentools mit Go: Daten weiterleiten. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!