A.31. Channel

Channel digunakan untuk menghubungkan goroutine satu dengan goroutine lain. Mekanisme yang dilakukan adalah serah-terima data lewat channel tersebut. Dalam komunikasinya, sebuah channel difungsikan sebagai pengirim di sebuah goroutine, dan juga sebagai penerima di goroutine lainnya. Pengiriman dan penerimaan data pada channel bersifat blocking atau synchronous.

Dasar Pemrograman Golang - Analogi channel

Pada chapter ini kita akan belajar mengenai pemanfaatan channel.

A.31.1. Penerapan Channel

Channel merupakan sebuah variabel, dibuat dengan menggunakan kombinasi keyword make dan chan. Variabel channel memiliki satu tugas, menjadi pengirim, atau penerima data.

Program berikut adalah contoh implementasi channel. 3 buah goroutine dieksekusi, di masing-masing goroutine terdapat proses pengiriman data lewat channel. Data tersebut akan diterima 3 kali di goroutine utama main.

package main

import "fmt"
import "runtime"

func main() {
    runtime.GOMAXPROCS(2)

    var messages = make(chan string)

    var sayHelloTo = func(who string) {
        var data = fmt.Sprintf("hello %s", who)
        messages <- data
    }

    go sayHelloTo("john wick")
    go sayHelloTo("ethan hunt")
    go sayHelloTo("jason bourne")

    var message1 = <-messages
    fmt.Println(message1)

    var message2 = <-messages
    fmt.Println(message2)

    var message3 = <-messages
    fmt.Println(message3)
}

Pada kode di atas, variabel messages dideklarasikan bertipe channel string. Cara pembuatan channel yaitu dengan menuliskan keyword make dengan isi keyword chan diikuti dengan tipe data channel yang diinginkan.

var messages = make(chan string)

Selain itu disiapkan juga closure sayHelloTo yang tugasnya membuat sebuah pesan string yang kemudian dikirim via channel. Pesan string tersebut dikirim lewat channel messages. Tanda <- jika dituliskan di sebelah kiri nama variabel, berarti sedang berlangsung proses pengiriman data dari variabel yang berada di kanan lewat channel yang berada di kiri (pada konteks ini, variabel data dikirim lewat channel messages).

var sayHelloTo = func(who string) {
    var data = fmt.Sprintf("hello %s", who)
    messages <- data
}

Fungsi sayHelloTo dieksekusi tiga kali sebagai goroutine berbeda. Menjadikan tiga proses ini berjalan secara asynchronous atau tidak saling tunggu.

Sekali lagi perlu diingat bahwa eksekusi goroutine adalah asynchronous, sedangkan serah-terima data antar channel adalah synchronous.

go sayHelloTo("john wick")
go sayHelloTo("ethan hunt")
go sayHelloTo("jason bourne")

Dari ketiga fungsi tersebut, goroutine yang selesai paling awal akan mengirim data lebih dulu, datanya kemudian diterima variabel message1. Tanda <- jika dituliskan di sebelah kiri channel, menandakan proses penerimaan data dari channel yang di kanan, untuk disimpan ke variabel yang di kiri.

var message1 = <-messages
fmt.Println(message1)

Penerimaan channel bersifat blocking. Artinya statement var message1 = <-messages hingga setelahnya tidak akan dieksekusi sebelum ada data yang dikirim lewat channel.

Ke semua data yang dikirim dari tiga goroutine berbeda tersebut datanya akan diterima secara berurutan oleh message1, message2, message3; untuk kemudian ditampilkan.

Dasar Pemrograman Golang - Implementasi channel

Dari screenshot output di atas bisa dilihat bahwa text yang dikembalikan oleh sayHelloTo tidak selalu berurutan, meskipun penerimaan datanya adalah berurutan. Hal ini dikarenakan, pengiriman data adalah dari 3 goroutine yang berbeda, yang kita tidak tau mana yang prosesnya selesai lebih dulu. Goroutine yang dieksekusi lebih awal belum tentu selesai lebih awal, yang jelas proses yang selesai lebih awal datanya akan diterima lebih awal.

Karena pengiriman dan penerimaan data lewat channel bersifat blocking, tidak perlu memanfaatkan sifat blocking dari fungsi sejenis fmt.Scanln() atau lainnya, untuk mengantisipasi goroutine utama main selesai sebelum ketiga goroutine di atas selesai.

A.31.2. Channel Sebagai Tipe Data Parameter

Variabel channel bisa di-pass ke fungsi lain sebagai parameter. Cukup tambahkan keyword chan pada deklarasi parameter agar operasi pass channel variabel bisa dilakukan.

Siapkan fungsi printMessage dengan parameter adalah channel. Lalu ambil data yang dikirimkan lewat channel tersebut untuk ditampilkan.

func printMessage(what chan string) {
    fmt.Println(<-what)
}

Setelah itu ubah implementasi di fungsi main.

func main() {
    runtime.GOMAXPROCS(2)

    var messages = make(chan string)

    for _, each := range []string{"wick", "hunt", "bourne"} {
        go func(who string) {
            var data = fmt.Sprintf("hello %s", who)
            messages <- data
        }(each)
    }

    for i := 0; i < 3; i++ {
        printMessage(messages)
    }
}

Output program di atas sama dengan program sebelumnya.

Parameter what fungsi printMessage bertipe channel string, bisa dilihat dari kode chan string pada cara deklarasinya. Operasi serah-terima data akan bisa dilakukan pada variabel tersebut, dan akan berdampak juga pada variabel messages di fungsi main.

Passing data bertipe channel lewat parameter sifatnya pass by reference, yang di transferkan adalah pointer datanya, bukan nilai datanya.

Dasar Pemrograman Golang - Parameter channel


Berikut merupakan penjelasan tambahan untuk kode di atas.

• Iterasi Data Slice/Array Langsung Pada Saat Inisialisasi

Data slice yang baru di inisialisasi bisa langsung di-iterasi, caranya mudah dengan menuliskannya langsung setelah keyword range.

for _, each := range []string{"wick", "hunt", "bourne"} {
    // ...
}

• Eksekusi Goroutine Pada IIFE

Eksekusi goroutine tidak harus pada fungsi atau closure yang sudah terdefinisi. Sebuah IIFE juga bisa dijalankan sebagai goroutine baru. Caranya dengan langsung menambahkan keyword go pada waktu deklarasi-eksekusi IIFE-nya.

var messages = make(chan string)

go func(who string) {
    var data = fmt.Sprintf("hello %s", who)
    messages <- data
}("wick")

var message = <-messages
fmt.Println(message)