Heim  >  Artikel  >  Backend-Entwicklung  >  Befehlszeilentools mit Go: Daten weiterleiten

Befehlszeilentools mit Go: Daten weiterleiten

Barbara Streisand
Barbara StreisandOriginal
2024-09-25 06:28:01450Durchsuche

Command-Line Tools with Go: Piping Data

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.

Zeilennummern zur Ausgabe hinzufügen

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 }

Base64-Kodierungseingabe

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!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn