Generics ใน Go
Natcha Luangaroonchai
Generics เป็นหนึ่งในฟีเจอร์ที่นักพัฒนาเรียกร้องกันมา ตั้งแต่เปิดตัวภาษา และ เริ่มพัฒนาจริง ๆ ตอนปี 2019 ในที่สุดก็ปล่อยออกมาให้ใช้งานกันแล้วที่เวอร์ชัน 1.18 มาลองดูกันว่าสามารถทำอะไรได้บ้าง
TL;DR
Generics เป็นแนวคิดของภาษายุคใหม่ที่อนุญาตให้ฟังก์ชันสามารถทำงานกับตัวแปรประเภทใดก็ได้ ที่ตรงกับเงื่อนไขของฟังก์ชันนั้น ๆ
ยกตัวอย่างเช่นฟังก์ชัน sort.Sort ที่ทำงานได้แค่กับตัวแปรที่อิมพลิเมนต์อินเตอร์เฟซ sort.Interface เท่านั้นซึ่งถ้าต้องการเรียงลำดับตัวแปรประเภทอื่นต้องอิมพลิเมนต์ฟังก์ชันเพิ่มอีก 3 ฟังก์ชันคือ 
- Len()
- Less(i, j int) bool
- Swap(i, j int)
การมาของ generics ช่วยให้การเขียนฟังก์ชันเรียงลำดับสามารถส่งสไลด์ของ []float64, []int และ []string เข้าไปที่ฟังก์ชัน sortAny ได้ทันทีโดยไม่ต้องแปลงเป็น sort.Float64Slice, sort.IntSlice และ sort.StringSlice
package main
import (
        "fmt"
        "sort"
        "golang.org/x/exp/constraints"
)
func main() {
        f := []float64{4.2, 2.9, 5.32, 1.70, 3.65}
        sortAny(f)
        fmt.Println(f)
        i := []int{6, 5, 1, 3, 4, 2}
        sortAny(i)
        fmt.Println(i)
        s := []string{"He", "She", "They", "It", "We"}
        sortAny(s)
        fmt.Println(s)
}
func sortAny[T constraints.Ordered](t []T) {
        sort.Slice(t, func(i, j int) bool { return t[i] < t[j] })
}สังเกตที่ฟังก์ชัน sortAny จะมี syntax ใหม่เป็นวงเล็บสี่เหลี่ยม 
[T constraints.Ordered]ถ้ากดเข้าไปดูจะพบว่า constraints.Ordered ประกาศเป็นอินเตอร์เฟซประกอบไปด้วย Integer | Float | ~string
// Ordered is a constraint that permits any ordered type: any type
// that supports the operators < <= >= >.
// If future releases of Go add new ordered types,
// this constraint will be modified to include them.
type Ordered interface {
    Integer | Float | ~string
}การประกาศอินเตอร์เฟซแบบนี้ทำให้ฟังก์ชันที่รองรับ constraints.Ordered สามารถรับตัวแปร Integer, Float และ ~string ที่อยู่กายใต้ constraints.Ordered ได้เช่นกัน 
สิ่งที่ต้องรู้คือ Go ไม่อนุญาตให้ผสมประเภทตัวแปรได้ อย่างโค้ดด้านล่างนี้จะไม่สามารถคอมไพล์ได้เนื่องจากมีการผสมกันระหว่าง []float64 และ []int แม้ว่าทั้ง []float64 และ []int จะอยู่ภายใต้ constraints.Ordered เหมือนกันก็ตาม
func main() {
        f := []float64{4.2, 2.9, 5.32, 1.70, 3.65}
        sortAny(f)
        printAnySorted(f)
        i := []int{6, 5, 1, 3, 4, 2}
        sortAny(i)
        printAnySorted(i)
        s := []string{"He", "She", "They", "It", "We"}
        sortAny(s)
        printAnySorted(s)
        // error type []int of i does not match inferred type []float64 for []T
        printAnySorted(f, i)
}
func printAnySorted[T constraints.Ordered](t ...[]T) {
    for _, v := range t {
        fmt.Println(v)
    }
}สังเกตที่ ~string ความหมายของสัญลักษณ์ ~ คือนอกจาก string แล้วยังรวมไปถึงประเภทของตัวแปรที่ถูกสร้างขึ้นจาก string อีกทีเช่นกัน ตัวอย่าง MyString ด้านล่างนี้
func main() {
    ...
    u := []MyString{"C", "D", "B", "A", "E"}
    sortAny(u)
    fmt.Println(u)
}
type MyString stringนอกจากนี้ Go ยังเตรียมอินเตอร์เฟซบางส่วนสำหรับประเภทตัวแปรพื้นฐานมาให้แล้วอย่างเช่น comparable
func main() {
        ...
        fmt.Println(equal("hello", "world!"))
}
func equal[K comparable](i, j K) bool {
        return i == j
}หรือการประกาศ generics ที่ระดับ struct แบบนี้
func main() {
        ...
    n := node[string]{val: "test"}
    fmt.Println(n)
}
type node[T constraints.Ordered] struct {
    val T
}อ้างอิง