---
title: Go to Generics = Go to hell?
tags: Slide, Go, Generics Programming
description: View the slide with "Slide Mode".
slideOptions:
spotlight:
enabled: false
allottedMinutes: 25
---
<small>Go to Generics = Go to hell?</small>
===
<!-- .slide: data-background-color="pink" -->
<!-- .slide: data-transition="zoom" -->
What I experienced
after Generics Introduced into Go
> <small>intermediate :angry:</small>
> [name=郭學聰 Hsueh-Tsung Kuo]
> [time=Sun, 30 Jul 2023]
###### CC BY-SA 4.0
---
<!-- .slide: data-transition="convex" -->
## Who am I?
![fieliapm](https://www.gravatar.com/avatar/2aef78f04240a6ac9ccd473ba1cbd1e3?size=2048 =384x384)
<small>Someone (who?) said:
a game programmer should be able to draw cute anime character(?)</small>
----
<!-- .slide: data-transition="convex" -->
* A ~~programmer~~ coding peasant from game company in Taiwan.
* Backend (and temporary frontend) engineer.
* Usually develop something related to my work in Python, Ruby, ECMAScript, Golang, C#.
* Built CDN-aware game asset update system.
* Business large passenger vehicle driver. :bus:
* Ride bike to help traffic jam. :racing_motorcycle:
* Care about chaotic traffic in Taiwan.
* Draw cute anime character in spare time.
---
<!-- .slide: data-transition="convex" -->
## Outline
----
<!-- .slide: data-transition="convex" -->
4. Introduction to Generics
* Languages
* Boxing & Monomorphization
5. Generics in Go
* Usage
* Type Constraint
* Mixed Constraint
----
<!-- .slide: data-transition="convex" -->
6. Pitfalls
* Usage
* Generic Methods
* Underlying Type
* Structure Members
* Performance
* Monomorphic vs Interface vs Generics
* GC Shape Stenciling
* Callbacks
----
<!-- .slide: data-transition="convex" -->
7. Possible Use Cases
* Collections
* Callback
* I/O of Data Structures
* Serializers
* Datastores
* Plugins
* Mocks
8. Conclusion
9. Resource
10. Q&A
---
<!-- .slide: data-transition="convex" -->
## Introduction to Generics
----
<!-- .slide: data-transition="convex" -->
### Generics
"algorithms are written in terms of data types to-be-specified-later that are then instantiated when needed for specific types provided as parameters."
----
<!-- .slide: data-transition="convex" -->
### Languages
C++
```cpp=
template <typename T>
T max (T a, T b) {
return a>b?a:b;
}
max(1, 2);
template<typename T>
class list {
public:
void append(T data) {
...
}
}
list<string> ls;
```
----
<!-- .slide: data-transition="convex" -->
### Languages
Java
```java=
public class List<T> {
public void append(T data) {
...
}
}
List<String> list = new List<String>();
```
----
<!-- .slide: data-transition="convex" -->
### Languages
Rust
```rust=
let mut language_codes: HashMap<&str, &str> = HashMap::new();
```
----
<!-- .slide: data-transition="convex" -->
### Boxing & Monomorphization
* Boxing
* run-time
* placing a primitive type within an object.
* Monomorphization
* compile-time
* polymorphic functions :arrow_right: monomorphic functions for each unique instantiation.
----
<!-- .slide: data-transition="convex" -->
#### Boxing
C#
```csharp=
int i = 300;
object o = i; // boxing
int j = (int)o; // unboxing
```
----
<!-- .slide: data-transition="convex" -->
#### Monomorphization
Rust
```rust=
fn id<T>(x: T) -> T {
return x;
}
fn main() {
let int = id(10);
let string = id("some text");
println!("{int}, {string}");
}
```
```rust=
fn id_i32(x: i32) -> i32 {
return x;
}
fn id_str(x: &str) -> &str {
return x;
}
fn main() {
let int = id_i32(10);
let string = id_str("some text");
println!("{int}, {string}");
}
```
---
<!-- .slide: data-transition="convex" -->
## Generics in Go
----
<!-- .slide: data-transition="convex" -->
### Usage
```go=
package main
import "fmt"
func MapKeys[K comparable, V any](m map[K]V) []K {
r := make([]K, 0, len(m))
for k := range m {
r = append(r, k)
}
return r
}
type List[T any] struct {
head, tail *element[T]
}
type element[T any] struct {
next *element[T]
val T
}
func (lst *List[T]) Push(v T) {
if lst.tail == nil {
lst.head = &element[T]{val: v}
lst.tail = lst.head
} else {
lst.tail.next = &element[T]{val: v}
lst.tail = lst.tail.next
}
}
func (lst *List[T]) GetAll() []T {
var elems []T
for e := lst.head; e != nil; e = e.next {
elems = append(elems, e.val)
}
return elems
}
func main() {
var m = map[int]string{1: "2", 2: "4", 4: "8"}
fmt.Println("keys:", MapKeys(m))
_ = MapKeys[int, string](m)
lst := List[int]{}
lst.Push(10)
lst.Push(13)
lst.Push(23)
fmt.Println("list:", lst.GetAll())
}
```
<small>https://gobyexample.com/generics</small>
----
<!-- .slide: data-transition="convex" -->
### Type Constraint
```go=
import "golang.org/x/exp/constraints"
func GMin[T constraints.Ordered](x, y T) T {
if x < y {
return x
}
return y
}
x := GMin[int](2, 3)
y := GMin(2, 3) // type inference
```
----
<!-- .slide: data-transition="convex" -->
### Type Constraint
```go=
type Signed interface {
~int | ~int8 | ~int16 | ~int32 | ~int64
}
type Unsigned interface {
~uint | ~uint8 | ~uint16 | ~uint32 | ~uint64 | ~uintptr
}
type Integer interface {
Signed | Unsigned
}
type Float interface {
~float32 | ~float64
}
type Ordered interface {
Integer | Float | ~string
}
```
----
<!-- .slide: data-transition="convex" -->
### Mixed Constraint
```go=
package main
import (
"fmt"
)
type MyType[T any] interface {
MyType1 | MyType2
// and
Concat(T) T
}
type MyType1 struct {
Str string
}
func (mt MyType1) Concat(s string) string {
fmt.Println("Concat MyType1")
return mt.Str + s
}
type MyType2 struct {
Int int
}
func (mt MyType2) Concat(i int) int {
fmt.Println("Concat MyType2")
return mt.Int + i
}
func ConcatMyType[T ~string | ~int, MT MyType[T]](mt MT, v T) T {
return mt.Concat(v)
}
func main() {
mt1 := MyType1{Str: "Sak"}
mt2 := MyType2{Int: 30}
fmt.Println(ConcatMyType[string](mt1, "ura"))
fmt.Println(ConcatMyType[int](mt2, 5))
// type inference
fmt.Println(ConcatMyType(mt1, "ura"))
fmt.Println(ConcatMyType(mt2, 5))
}
```
<small>好油喔peko :rabbit:</small> <!-- .element: class="fragment" data-fragment-index="1" -->
---
<!-- .slide: data-transition="convex" -->
## Pitfalls
----
<!-- .slide: data-transition="convex" -->
### Usage
----
<!-- .slide: data-transition="convex" -->
#### Generic Methods
```go=
package main
import (
"fmt"
"golang.org/x/exp/constraints"
)
type List[T constraints.Ordered] struct {
list []T
}
func (ls *List[T]) Append(v T) {
ls.list = append(ls.list, v)
}
func (ls *List[T]) All() []T {
return ls.list
}
func (ls *List[T]) String() string {
return fmt.Sprintf("%v", ls.list)
}
// syntax error: method must have no type parameters
func (ls *List[T]) String[R ~string | ~[]byte]() R {
return fmt.Sprintf("%v", ls.list)
}
func ToString[T constraints.Ordered, R ~string | ~[]byte](ls List[T]) R {
return R(fmt.Sprintf("%v", ls.list))
}
func main() {
var ls List[int]
ls.Append(1)
ls.Append(2)
ls.Append(3)
fmt.Println(ls.All())
fmt.Println(ls.String())
fmt.Println(ToString[int, []byte](ls))
}
```
----
<!-- .slide: data-transition="convex" -->
#### Underlying Type
```go=
package main
type Type struct{}
type MyType Type
func F[T ~struct{}](t T) {}
// invalid use of ~ (underlying type of Type is struct{})
func F[T ~Type](t T) {}
func main() {
x := Type{}
y := MyType{}
F(x)
F(y)
}
```
----
<!-- .slide: data-transition="convex" -->
#### Structure Members
```go=
package main
import "fmt"
type Type struct {
Str string
Int int
}
type MyType Type
func F[T ~struct {
Str string
Int int
}](t T) string {
// t.Str undefined (type T has no field or method Str)
// t.Int undefined (type T has no field or method Int)
return fmt.Sprintf("%v %v", t.Str, t.Int)
}
func main() {
x := Type{}
y := MyType{}
fmt.Println(F(x))
fmt.Println(F(y))
}
```
but permit to access interface methods <!-- .element: class="fragment" data-fragment-index="1" -->
----
<!-- .slide: data-transition="convex" -->
### Performance
go 1.18~1.20
----
<!-- .slide: data-transition="convex" -->
#### Monomorphic vs Interface vs Generics
```go=
package main
import (
"bytes"
"fmt"
"io"
"time"
)
type Data struct {
Name string
Value string
}
func SerializeM(w *bytes.Buffer, d Data) {
b := []byte(d.Name + " " + d.Value)
w.Write(b)
}
func SerializeI(w io.Writer, d Data) {
b := []byte(d.Name + " " + d.Value)
w.Write(b)
}
func SerializeG[W io.Writer](w W, d Data) {
b := []byte(d.Name + " " + d.Value)
w.Write(b)
}
func main() {
var buf bytes.Buffer
var bw io.ReadWriter
bw = &buf
data := Data{Name: "Sakura", Value: "35"}
startTime := time.Now()
for i := 0; i < 100000; i++ {
//SerializeM(&buf, data) // 3.614264ms
//SerializeI(&buf, data) // 3.667853ms
//SerializeG(&buf, data) // 6.934544ms
SerializeG(bw, data) // 7.680705ms
}
endTime := time.Now()
fmt.Println(endTime.Sub(startTime))
}
```
----
<!-- .slide: data-transition="convex" -->
#### Monomorphic vs Interface vs Generics
Monomorphic
```go=
"".SerializeM STEXT size=197 args=0x28 locals=0x88 funcid=0x0 align=0x0
0x0065 00101 (main.go:17) MOVQ CX, DI
0x0068 00104 (main.go:17) MOVQ BX, CX
0x006b 00107 (main.go:17) MOVQ AX, BX
0x006e 00110 (main.go:17) MOVQ ""..autotmp_9+120(SP), AX
0x0073 00115 (main.go:17) PCDATA $1, $2
0x0073 00115 (main.go:17) CALL bytes.(*Buffer).Write(SB)
```
----
<!-- .slide: data-transition="convex" -->
#### Monomorphic vs Interface vs Generics
Interface
```go=
"".SerializeI STEXT size=200 args=0x30 locals=0x60 funcid=0x0 align=0x0
0x0021 00033 (main.go:22) MOVQ AX, "".w+104(SP)
0x0026 00038 (main.go:22) MOVQ BX, "".w+112(SP)
0x002b 00043 (main.go:22) PCDATA $3, $2
0x005a 00090 (main.go:22) MOVQ "".w+104(SP), DX
0x005f 00095 (main.go:22) MOVQ 24(DX), DX
0x0063 00099 (main.go:22) MOVQ CX, DI
0x0066 00102 (main.go:22) MOVQ BX, CX
0x0069 00105 (main.go:22) MOVQ AX, BX
0x006c 00108 (main.go:22) MOVQ "".w+112(SP), AX
0x0071 00113 (main.go:22) PCDATA $1, $2
0x0071 00113 (main.go:22) CALL DX
```
----
<!-- .slide: data-transition="convex" -->
#### Monomorphic vs Interface vs Generics
Generics (pointer arg)
```go=
"".SerializeG[go.shape.*uint8_0] STEXT dupok size=200 args=0x30 locals=0x60 funcid=0x0 align=0x0
0x005a 00090 (main.go:27) PCDATA $0, $-2
0x005a 00090 (main.go:27) MOVQ ""..dict+104(SP), DX
0x005f 00095 (main.go:27) PCDATA $0, $-1
0x005f 00095 (main.go:27) MOVQ 16(DX), DX
0x0063 00099 (main.go:27) MOVQ 24(DX), DX
0x0067 00103 (main.go:27) MOVQ CX, DI
0x006a 00106 (main.go:27) MOVQ BX, CX
0x006d 00109 (main.go:27) MOVQ AX, BX
0x0070 00112 (main.go:27) MOVQ "".w+112(SP), AX
0x0075 00117 (main.go:27) PCDATA $1, $2
0x0075 00117 (main.go:27) CALL DX
```
----
<!-- .slide: data-transition="convex" -->
#### Monomorphic vs Interface vs Generics
Generics (interface arg)
```go=
"".main STEXT size=395 args=0x0 locals=0xb0 funcid=0x0 align=0x0
0x00b3 00179 (main.go:27) LEAQ go.itab.*bytes.Buffer,io.ReadWriter(SB), BX
"".SerializeG[go.shape.interface { Read([]uint8) (int, error); Write([]uint8) (int, error) }_0] STEXT dupok size=253 args=0x38 locals=0x80 funcid=0x0 align=0x0
0x0030 00048 (main.go:27) MOVQ CX, "".w+152(SP)
0x0038 00056 (main.go:27) PCDATA $3, $2
0x0038 00056 (main.go:27) MOVQ BX, ""..autotmp_10+112(SP)
0x0075 00117 (main.go:27) LEAQ type.io.Writer(SB), AX
0x007c 00124 (main.go:27) MOVQ ""..autotmp_10+112(SP), BX
0x0081 00129 (main.go:27) PCDATA $1, $2
0x0081 00129 (main.go:27) CALL runtime.assertI2I(SB)
0x0086 00134 (main.go:27) MOVQ 24(AX), DX
0x008a 00138 (main.go:27) MOVQ "".w+152(SP), AX
0x0092 00146 (main.go:27) MOVQ "".b.ptr+104(SP), BX
0x0097 00151 (main.go:27) MOVQ "".b.len+56(SP), CX
0x009c 00156 (main.go:27) MOVQ "".b.cap+64(SP), DI
0x00a1 00161 (main.go:27) PCDATA $1, $3
0x00a1 00161 (main.go:27) CALL DX
```
----
<!-- .slide: data-transition="convex" -->
##### GC Shape Stenciling
* Stencil the code for each different GC shape of the instantiated types.
* Using a dictionary to handle differing behaviors of types that have the same shape.
* `*AnyType` :arrow_right: `SerializeG[go.shape.*uint8_0]`
* `Interface` :arrow_right: `SerializeG[go.shape.interface { ... }_0]`
----
<!-- .slide: data-transition="convex" -->
##### Interface Table
```go=
type iface struct {
tab *itab
data unsafe.Pointer
}
type itab struct {
inter *interfacetype // offset 0
_type *_type // offset 8
hash uint32 // offset 16
_ [4]byte
fun [1]uintptr // offset 24...
}
```
----
<!-- .slide: data-transition="convex" -->
##### :shit: :shit: :shit:
pointer arg :arrow_right: `dict` + `iface.itab.fun` :arrow_right: :shit:
interface arg :arrow_right: `runtime.assertI2I()` + `iface.itab.fun` :arrow_right: :shit:
----
<!-- .slide: data-transition="convex" -->
##### Exception
`string` :arrow_right: `SequenceToString[go.shape.string_0]`
`[]byte` :arrow_right: `SequenceToString[go.shape.[]uint8_0]`
```go=
type Byteseq interface {
~string | ~[]byte
}
func SequenceToString[T Byteseq](x T) string {
ix := any(x)
switch ix.(type) {
case string:
return ix.(string)
case []byte:
p := ix.([]byte)
return *(*string)(unsafe.Pointer(&p))
default:
return ""
}
}
```
<small>https://github.com/koykov/byteseq</small>
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
----
<!-- .slide: data-transition="convex" -->
##### Callbacks
plain
```go=
package main
import "fmt"
func MapInt(a []int, callback func(int) int) []int {
for n, elem := range a {
a[n] = callback(elem)
}
return a
}
func main() {
input := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
output := MapInt(input, func(i int) int {
return i + 7
})
fmt.Println(output)
}
```
----
<!-- .slide: data-transition="convex" -->
##### Callbacks
plain (inlined)
```go=
0x0074 00116 (main.go:14) XORL CX, CX
0x0076 00118 (main.go:6) JMP 128
0x0078 00120 (main.go:7) ADDQ $7, (AX)(CX*8)
0x007d 00125 (main.go:6) INCQ CX
0x0080 00128 (main.go:6) CMPQ CX, $10
0x0084 00132 (main.go:6) JLT 120
"".MapInt STEXT size=170 args=0x20 locals=0x20 funcid=0x0 align=0x0
```
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
parametrize callback parameter
```go=
package main
import "fmt"
func MapAny[I any](a []I, callback func(I) I) []I {
for n, elem := range a {
a[n] = callback(elem)
}
return a
}
func main() {
input := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
output := MapAny(input, func(i int) int {
return i + 7
})
fmt.Println(output)
}
```
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
parametrize callback parameter (not inlined)
```go=
0x0079 00121 (main.go:14) XORL CX, CX
0x007b 00123 (main.go:6) JMP 173
0x007d 00125 (main.go:6) MOVQ CX, "".n+40(SP)
0x0082 00130 (main.go:6) MOVQ (AX)(CX*8), BX
0x0086 00134 (main.go:7) MOVQ "".main.func1·f(SB), SI
0x008d 00141 (main.go:7) LEAQ "".main.func1·f(SB), DX
0x0094 00148 (main.go:7) MOVQ BX, AX
0x0097 00151 (main.go:7) PCDATA $1, $1
0x0097 00151 (main.go:7) CALL SI
0x0099 00153 (main.go:7) MOVQ "".n+40(SP), CX
0x009e 00158 (main.go:7) MOVQ ""..autotmp_38+48(SP), BX
0x00a3 00163 (main.go:7) MOVQ AX, (BX)(CX*8)
0x00a7 00167 (main.go:6) INCQ CX
0x00aa 00170 (main.go:6) MOVQ BX, AX
0x00ad 00173 (main.go:6) CMPQ CX, $10
0x00b1 00177 (main.go:6) JLT 125
"".MapAny[go.shape.int_0] STEXT dupok size=186 args=0x28 locals=0x20 funcid=0x0 align=0x0
```
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
parametrize callback
```go=
package main
import "fmt"
func MapInt[F func(int) int](a []int, callback F) []int {
for n, elem := range a {
a[n] = callback(elem)
}
return a
}
func main() {
input := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
output := MapInt(input, func(i int) int {
return i + 7
})
fmt.Println(output)
}
```
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
parametrize callback (inlined)
```go=
0x0074 00116 (main.go:14) XORL CX, CX
0x0076 00118 (main.go:6) JMP 128
0x0078 00120 (main.go:7) ADDQ $7, (AX)(CX*8)
0x007d 00125 (main.go:6) INCQ CX
0x0080 00128 (main.go:6) CMPQ CX, $10
0x0084 00132 (main.go:6) JLT 120
"".MapInt[go.shape.func(int) int_0] STEXT dupok size=186 args=0x28 locals=0x20 funcid=0x0 align=0x0
```
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
parametrize callback and its parameter
```go=
package main
import "fmt"
func MapAny[I any, F func(I) I](a []I, callback F) []I {
for n, elem := range a {
a[n] = callback(elem)
}
return a
}
func main() {
input := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
output := MapAny(input, func(i int) int {
return i + 7
})
fmt.Println(output)
}
```
----
<!-- .slide: data-transition="convex" -->
#### Callbacks
parametrize callback and its parameter (inlined)
```go=
0x0074 00116 (main.go:14) XORL CX, CX
0x0076 00118 (main.go:6) JMP 135
0x0078 00120 (main.go:6) MOVQ (AX)(CX*8), DX
0x007c 00124 (<unknown line number>) NOP
0x007c 00124 (main.go:15) ADDQ $7, DX
0x0080 00128 (main.go:7) MOVQ DX, (AX)(CX*8)
0x0084 00132 (main.go:6) INCQ CX
0x0087 00135 (main.go:6) CMPQ CX, $10
0x008b 00139 (main.go:6) JLT 120
"".MapAny[go.shape.int_0,go.shape.func(int) int_1] STEXT dupok size=186 args=0x28 locals=0x20 funcid=0x0 align=0x0
```
---
<!-- .slide: data-transition="convex" -->
## Possible Use Cases
----
<!-- .slide: data-transition="convex" -->
### Collections
```go=
type List[T constraints.Ordered] struct {
list []T
}
func (ls *List[T]) Append(v T) {
ls.list = append(ls.list, v)
}
func (ls *List[T]) All() []T {
return ls.list
}
```
good practice :o:
- T is pointer or interface :x:
----
<!-- .slide: data-transition="convex" -->
### Callback
```go=
package main
import "fmt"
func MapAny[I any, F func(I) I](a []I, callback F) []I {
for n, elem := range a {
a[n] = callback(elem)
}
return a
}
func main() {
input := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
output := MapAny(input, func(i int) int {
return i + 7
})
fmt.Println(output)
}
```
good practice :o:
- do parametrize callback type :warning:
----
<!-- .slide: data-transition="convex" -->
### I/O of Data Structures
----
<!-- .slide: data-transition="convex" -->
#### Serializers
```go=
func SerializeG[W io.Writer](w W, d Data) {
b := []byte(d.Name + " " + d.Value)
w.Write(b)
}
```
bad practice :x:
- performance down
----
<!-- .slide: data-transition="convex" -->
#### Datastores
```go=
func SerializeG[W io.Writer](w W, d Data) {
b := []byte(d.Name + " " + d.Value)
w.Write(b)
}
func Datastore(d Data) {
SerializeG[TCPWriter](tcpW, d)
}
```
bad practice :x:
- performance down
- I/O latency >>>>>> code performance
----
<!-- .slide: data-transition="convex" -->
### Plugins
```go=
type Plugin[T any] interface {
Method(T) T
}
```
acceptable? :thinking_face:
- T is pointer or interface :x:
- however...
----
<!-- .slide: data-transition="convex" -->
### Mocks
```go=
type Foo interface {
Bar(x int) int
}
func SUT(f Foo) {
// ...
}
func TestFoo(t *testing.T) {
ctrl := gomock.NewController(t)
// Assert that Bar() is invoked.
defer ctrl.Finish()
m := NewMockFoo(ctrl)
// Asserts that the first and only call to Bar() is passed 99.
// Anything else will fail.
m.
EXPECT().
Bar(gomock.Eq(99)).
Return(101)
SUT(m)
}
```
practically impossible :x:
- not working in many cases :-1:
---
<!-- .slide: data-transition="convex" -->
## Conclusion
----
<!-- .slide: data-transition="convex" -->
### Prefer
* Use generics in data structures.
* Collections
* Vectors & Matrices & Tensors
* Pass value types to generic functions.
* Parametrize callback types.
* Parametrize constraint:
* `type Byteseq interface{ ~string | ~[]byte }`
----
<!-- .slide: data-transition="convex" -->
### Forbidden
* :x: Parametrize any method parameter.
* :x: Attempt to pass any pointer arg :arrow_right: expect generic function monomorphized & inlined.
* Twice dereference when calling interface method.
* :no_entry: Pass interface arg to generic function.
* Convert interface then dereference when calling interface method.
----
<!-- .slide: data-transition="convex" -->
### Slogan
:hash: {善用泛型,不濫用泛型|<big>Make good use of generics, don't abuse generics.</big>}
> [name=郭學聰 Hsueh-Tsung Kuo] [time=2023_07_30] [color=red] :notebook:
---
<!-- .slide: data-transition="convex" -->
## Resource
----
<!-- .slide: data-transition="convex" -->
### Reference
* Generics can make your Go code slower[^ref1]
* An Introduction To Generics[^ref2]
* When To Use Generics[^ref3]
* No parameterized methods[^ref4]
[^ref1]:<small>https://planetscale.com/blog/generics-can-make-your-go-code-slower</small>
[^ref2]:<small>https://go.dev/blog/intro-generics</small>
[^ref3]:<small>https://go.dev/blog/when-generics</small>
[^ref4]:<small>https://go.googlesource.com/proposal/+/refs/heads/master/design/43651-type-parameters.md#No-parameterized-methods</small>
---
<!-- .slide: data-transition="zoom" -->
## Q&A
---
<style>
.reveal {
background: #FFDFEF;
color: black;
}
.reveal h2,
.reveal h3,
.reveal h4,
.reveal h5,
.reveal h6 {
color: black;
}
.reveal code {
font-size: 18px !important;
line-height: 1.2;
}
.progress div{
height:14px !important;
background: hotpink !important;
}
// original template
.rightpart{
float:right;
width:50%;
}
.leftpart{
margin-right: 50% !important;
height:50%;
}
.reveal section img { background:none; border:none; box-shadow:none; }
p.blo {
font-size: 50px !important;
background:#B6BDBB;
border:1px solid silver;
display:inline-block;
padding:0.5em 0.75em;
border-radius: 10px;
box-shadow: 5px 5px 5px #666;
}
p.blo1 {
background: #c7c2bb;
}
p.blo2 {
background: #b8c0c8;
}
p.blo3 {
background: #c7cedd;
}
p.bloT {
font-size: 60px !important;
background:#B6BDD3;
border:1px solid silver;
display:inline-block;
padding:0.5em 0.75em;
border-radius: 8px;
box-shadow: 1px 2px 5px #333;
}
p.bloA {
background: #B6BDE3;
}
p.bloB {
background: #E3BDB3;
}
/*.slide-number{
margin-bottom:10px !important;
width:100%;
text-align:center;
font-size:25px !important;
background-color:transparent !important;
}*/
iframe.myclass{
width:100px;
height:100px;
bottom:0;
left:0;
position:fixed;
border:none;
z-index:99999;
}
h1.raw {
color: #fff;
background-image: linear-gradient(90deg,#f35626,#feab3a);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
animation: hue 5s infinite linear;
}
@keyframes hue {
from {
filter: hue-rotate(0deg);
}
to {
filter: hue-rotate(360deg);
}
}
.progress{
height:14px !important;
}
.progress span{
height:14px !important;
background: url("") repeat-x !important;
}
.progress span:after,
.progress span.nyancat{
content: "";
background: url('') !important;
width: 34px !important;
height: 21px !important;
border: none !important;
float:right;
margin-top:-7px;
margin-right:-10px;
}
</style>