Handy Golang Cheatsheet For Quick Reference

Download PDF

Go, also known as Golang, is a statically typed, compiled programming language designed for simplicity and efficiency. Developed by Google, Go emphasizes readability, conciseness, and robustness. With a rich standard library and support for concurrent programming, Go excels in building scalable, high-performance applications for web development, cloud computing, and system programming. Its features include goroutines for lightweight concurrency, channels for communication between goroutines, and interfaces for polymorphism. Go’s fast compilation and execution, along with its strong focus on simplicity and efficiency, make it a popular choice for modern software development across various domains.

{{< toc >}}

Hello world in Go

package main

import "fmt"

func main() {
    fmt.Println("Hello, World!")
}

Output:

Hello, World!

Variables in go

In Go, variables are declared using the var keyword followed by the variable name and type (optional). Here’s a basic example:

func main() {
  // Declaring variables
    var age int
    var name string = "John"
    var temperature float64 = 23.5

    // Short variable declaration (Type inference)
    var country string = "USA"
    var height float64 = 175.5

    // Assigning values to variables
    age = 30

    // Multiple variables
    var (
        a int
        b int
        c int
    )

    var (
        x int
        y = 20
        z int = 30
        d int
        e string = "Hello"
        f string
        g string
    )
}

Datatypes in Go

In Go, data types are used to define the type of data that a variable can hold. Here are the basic data types in Go along with examples:

Numeric Types:

  • int: Represents integer numbers.
    go
    var age int = 30

    int8, int16, int32, int64: Signed integers with specific bit sizes (8, 16, 32, 64 bits).
    uint: Represents unsigned integer numbers.
    uint8, uint16, uint32, uint64: Unsigned integers with specific bit sizes (8, 16, 32, 64 bits).
    float32, float64: Represents floating-point numbers.
    go
    var temperature float64 = 23.5
  • complex64, complex128: Represents complex numbers.
    go
    var z complex128 = complex(1, 2)

    ### Boolean Type:
    bool: Represents boolean values (true or false).
    go
    var isSunny bool = true

String Type:

  • string: Represents a sequence of characters.

     var name string = "John"
    

    Derived Types:

    • array: Represents a fixed-size collection of elements of the same type.
      go
      var numbers [5]int = [5]int{1, 2, 3, 4, 5}
  • slice: Represents a dynamic-size collection of elements of the same type.

     var fruits []string = []string{"apple", "banana", "orange"}
    
    • map: Represents a collection of key-value pairs.
      go
      var ages map[string]int = map[string]int{"John": 30, "Alice": 25}
  • struct: Represents a collection of fields.

     type Person struct {
         Name string
         Age  int
     }
    
    
     var john Person = Person{Name: "John", Age: 30}
    

    Pointer Type:

    • pointer: Represents the memory address of a variable.
      go
      var ptr *int = &age

Function Type:

  • func: Represents a function.
    go
    func add(a, b int) int {
    return a + b
    }

    ### Interface Type:
    interface: Represents a set of method signatures.
    go
    type Shape interface {
    Area() float64
    Perimeter() float64
    }

Channel Type:

  • channel: Represents a communication channel between goroutines.
    go
    var ch chan int = make(chan int)

    {{< notice “note” >}}
    These are some of the basic data types in Go along with examples. Go also provides type conversion functions (int(), float64(), string(), etc.) to convert between different data types when needed.
    {{< /notice >}}
    ## Flow control in Go
    Overview of flow control in Go along with examples for each type:
    ### Conditional Statements:
    #### if Statement:
    The if statement is used for conditional execution of code blocks.
    go
    func main() {
    x := 10
    if x > 5 {
    fmt.Println("x is greater than 5")
    } else {
    fmt.Println("x is less than or equal to 5")
    }
    }

switch Statement:

The switch statement is used to make decisions based on the value of an expression.


func main() {
    fruit := "apple"
    switch fruit {
    case "apple":
        fmt.Println("Apple")
    case "banana":
        fmt.Println("Banana")
    default:
        fmt.Println("Unknown fruit")
    }
}

for Loop:

The for loop is used to repeatedly execute a block of code.


func main() {
    for i := 0; i < 5; i++ {
        fmt.Println(i)
    }
}

for Loop (while style):

Go doesn’t have a while loop, but you can achieve the same behavior using a for loop with a condition.


func main() {
    x := 0
    for x < 5 {
        fmt.Println(x)
        x++
    }
}

forrange Loop:

The forrange loop is used to iterate over elements in arrays, slices, maps, or strings.

func main() {
    numbers := []int{1, 2, 3, 4, 5}
    for index, value := range numbers {
        fmt.Printf("Index: %d, Value: %d\n", index, value)
    }
}

{{< notice “note” >}}

  • Go provides various flow control mechanisms such as if, switch, for, and forrange.
  • These constructs allow you to make decisions, iterate over collections, and control the flow of execution in your Go programs.
    {{< /notice >}}

Functions in Go

Overview of functions in Go covering lambdas, multiple return types, and named return values:

Lambdas (Anonymous Functions):

In Go, anonymous functions, also known as lambda functions, can be declared inline without a name. They are often used for short, one-off functions.

func main() {
    // Lambda function to add two numbers
    add := func(a, b int) int {
        return a + b
    }

    result := add(3, 5)
    fmt.Println("Result:", result) // Output: 8
}

Multiple Return Types:

Go supports returning multiple values from a function, which is commonly used for error handling or returning multiple results.


// Function to calculate sum and difference of two numbers
func calculate(a, b int) (int, int) {
    sum := a + b
    diff := a - b
    return sum, diff
}

func main() {
    sum, diff := calculate(10, 5)
    fmt.Println("Sum:", sum)   // Output: 15
    fmt.Println("Diff:", diff) // Output: 5
}

Named Return Values:

In Go, you can specify names for the return values of a function. Named return values are initialized to their zero values and can be modified in the function body. They are particularly useful for improving code readability.


// Function to divide two numbers and return quotient and remainder
func divide(a, b int) (quotient, remainder int) {
    quotient = a / b
    remainder = a % b
    return // Returns named return values
}

func main() {
    q, r := divide(10, 3)
    fmt.Println("Quotient:", q)   // Output: 3
    fmt.Println("Remainder:", r)  // Output: 1
}

{{< notice “note” >}}

  • Go supports lambdas, or anonymous functions, which are functions without a name.
  • Functions in Go can return multiple values, which is useful for various scenarios such as error handling.
  • Named return values in Go allow you to specify names for return values, improving code readability and enabling direct assignment of values in the function body.
    {{< /notice >}}

Packages in Go

Overview of packages in Go covering importing, aliases, packages, and exporting names:

Importing Packages:

In Go, you can import packages using the import keyword followed by the package path. You can import multiple packages in a single import statement, and you can also use blank identifiers _ to import packages for their side effects only.

package main

import (
    "fmt"
    "math/rand"
    "time"
)

Aliases:

You can provide an alias for an imported package using the import statement to avoid name conflicts or to make the code more readable.

package main

import (
    "fmt"
    myrand "math/rand"
)

Packages:

In Go, a package is a collection of Go source files that reside in the same directory. You can think of a package as a namespace that organizes a set of related functions, types, and variables.

package mypackage

func Add(a, b int) int {
    return a + b
}

Exporting Names:

In Go, names that start with a capital letter are exported from the package and can be accessed by other packages. Names that start with a lowercase letter are not exported and are only accessible within the package.

package mypackage

// Exported function
func Add(a, b int) int {
    return a + b
}

// Not exported (only accessible within the package)
func multiply(a, b int) int {
    return a * b
}

{{< notice “note” >}}

  • Go allows you to import packages using the import keyword followed by the package path.
  • You can provide an alias for an imported package to avoid name conflicts or improve readability.
  • Packages in Go organize related functions, types, and variables into namespaces.
  • Exported names in Go start with a capital letter and are accessible by other packages, while names starting with a lowercase letter are not exported and are only accessible within the package.
    {{< /notice >}}

Concurrency in Go

Concurrency in Go is a powerful feature that allows you to write efficient and scalable programs. Here’s an overview of key concepts in concurrency in Go, including goroutines, buffered channels, closing channels, and wait groups:

Goroutines:

Goroutines are lightweight threads of execution that allow functions to run concurrently with other functions. You can create a new goroutine using the go keyword followed by a function call.

package main

import (
    "fmt"
    "time"
)

func sayHello() {
    fmt.Println("Hello")
}

func main() {
    // Start a new goroutine
    go sayHello()

    // Wait for a moment to allow the goroutine to execute
    time.Sleep(100 * time.Millisecond)
}

Buffered Channels:

Channels are a built-in synchronization primitive in Go used for communication and synchronization between goroutines. Buffered channels have a capacity specified during creation, allowing them to hold a limited number of values without a receiver.

package main

import (
    "fmt"
)

func main() {
    // Create a buffered channel with capacity 3
    ch := make(chan int, 3)

    // Send values to the channel
    ch <- 1
    ch <- 2
    ch <- 3

    // Receive values from the channel
    fmt.Println(<-ch)
    fmt.Println(<-ch)
    fmt.Println(<-ch)
}

Closing Channels:

Closing a channel indicates that no more values will be sent on it. You can close a channel using the close function. A closed channel can still be read from, but it will always return the zero value immediately.

package main

import (
    "fmt"
)

func main() {
    ch := make(chan int)

    go func() {
        for i := 0; i < 5; i++ {
            ch <- i
        }
        close(ch) // Close the channel after sending all values
    }()

    for num := range ch {
        fmt.Println(num)
    }
}

WaitGroup:

The sync package provides the WaitGroup type, which allows you to wait for a collection of goroutines to finish. You can add goroutines to the WaitGroup using the Add method, and then wait for them to finish using the Wait method.

package main

import (
    "fmt"
    "sync"
)

func worker(id int, wg *sync.WaitGroup) {
    defer wg.Done() // Decrease the WaitGroup counter when the function finishes

    fmt.Printf("Worker %d starting\n", id)
    // Simulate some work
    time.Sleep(time.Second)
    fmt.Printf("Worker %d done\n", id)
}

func main() {
    var wg sync.WaitGroup

    for i := 1; i <= 3; i++ {
        wg.Add(1) // Increment the WaitGroup counter before starting a goroutine
        go worker(i, &wg)
    }

    wg.Wait() // Wait for all workers to finish
    fmt.Println("All workers done")
}

{{< notice “note” >}}

  • Goroutines are lightweight threads of execution in Go.
  • Channels facilitate communication and synchronization between goroutines.
  • Buffered channels have a capacity and can hold a limited number of values.
  • Closing a channel indicates that no more values will be sent on it.
  • WaitGroups allow you to wait for a collection of goroutines to finish before proceeding.
    {{< /notice >}}

Error control in Go

Error control in Go is crucial for writing robust and reliable code. Here’s an overview of how to handle errors and use the defer statement and deferring functions in Go:

Defer:

The defer statement is used to schedule a function call to be executed after the surrounding function returns. This can be useful for cleanup tasks or ensuring that resources are released, regardless of whether the function exits normally or with an error.

package main

import "fmt"

func main() {
    defer fmt.Println("Cleanup")
    fmt.Println("Hello, World!")
}

In this example, the Cleanup function will be called after the main function returns, ensuring that it is executed regardless of whether an error occurs or not.

Deferring Functions:

You can also defer the execution of a specific function call. This is particularly useful for closing resources such as files or database connections.

package main

import "fmt"

func cleanup() {
    fmt.Println("Cleanup")
}

func main() {
    defer cleanup()
    fmt.Println("Hello, World!")
}

In this example, the cleanup function will be called after the main function returns, ensuring that resources are properly cleaned up.

Error Handling:

Error handling in Go typically involves returning an error value from a function and checking for errors at the caller’s side using the if err != nil pattern.

package main

import (
    "errors"
    "fmt"
)

func divide(a, b int) (int, error) {
    if b == 0 {
        return 0, errors.New("division by zero")
    }
    return a / b, nil
}

func main() {
    result, err := divide(10, 0)
    if err != nil {
        fmt.Println("Error:", err)
        return
    }
    fmt.Println("Result:", result)
}

In this example, the divide function returns an error if the divisor is zero. The caller checks for errors using if err != nil and handles them accordingly.

{{< notice “note” >}}
– Error control in Go is essential for writing robust and reliable code.
– The defer statement is used to schedule a function call to be executed after the surrounding function returns.
– Deferring functions can be used to ensure cleanup tasks or resource releases.
– Error handling in Go typically involves returning an error value from a function and checking for errors at the caller’s side using the if err != nil pattern.
{{< /notice >}}

Structs in Go

Structs in Go are composite data types that allow you to group together different types of data under a single name. Here’s an overview of defining structs, using struct literals, and pointers to structs in Go:

Defining Structs:

You define a struct using the type keyword followed by the name of the struct and a set of field declarations within curly braces {}.


// Define a struct named Person with fields Name and Age
type Person struct {
    Name string
    Age  int
}

func main() {
    // Create a variable of type Person and initialize its fields
    var p1 Person
    p1.Name = "Alice"
    p1.Age = 30

    // Accessing struct fields
    fmt.Println("Name:", p1.Name)
    fmt.Println("Age:", p1.Age)
}

Struct Literals:

You can initialize a struct using a struct literal, which specifies the values of its fields.


type Point struct {
    X, Y int
}

func main() {
    // Create a Point struct using a struct literal
    p := Point{X: 10, Y: 20}

    fmt.Println("X coordinate:", p.X)
    fmt.Println("Y coordinate:", p.Y)
}

Pointers to Structs:

You can also use pointers to structs, which allow you to modify the struct’s fields directly.


type Person struct {
    Name string
    Age  int
}

func main() {
    // Create a pointer to a Person struct using the address-of operator (&)
    p := &Person{Name: "Bob", Age: 25}

    // Accessing and modifying struct fields using pointers
    fmt.Println("Name:", p.Name)
    fmt.Println("Age:", p.Age)

    p.Age = 30
    fmt.Println("Updated Age:", p.Age)
}

{{< notice “note” >}}
– Structs in Go allow you to group together different types of data under a single name.
– You define structs using the type keyword followed by the name of the struct and a set of field declarations within curly braces {}.
– Struct literals are used to initialize structs with specific values for their fields.
– Pointers to structs allow you to modify the struct’s fields directly and are created using the address-of operator &.
{{< /notice >}}

Methods in Go

Methods are functions associated with a particular type. Here’s an overview of methods in Go, including receivers and mutation:

Receivers:

A receiver is a parameter in a method declaration that associates the method with a specific type. Methods with receivers are called on instances of that type.


// Define a struct type
type Rectangle struct {
    Width, Height float64
}

// Method with a receiver of type Rectangle
func (r Rectangle) Area() float64 {
    return r.Width * r.Height
}

func main() {
    // Create an instance of Rectangle
    rect := Rectangle{Width: 10, Height: 5}

    // Call the Area method on the Rectangle instance
    fmt.Println("Area:", rect.Area()) // Output: 50
}

In this example, the Area method is associated with the Rectangle type through the receiver (r Rectangle). When you call rect.Area(), Go automatically passes rect as the receiver to the Area method.

Mutation:

Methods in Go can mutate the receiver if the receiver is a pointer type. This allows methods to modify the state of the receiver.


type Counter struct {
    Count int
}

// Method with a pointer receiver
func (c *Counter) Increment() {
    c.Count++
}

func main() {
    // Create an instance of Counter
    counter := Counter{Count: 0}

    // Call the Increment method, which mutates the receiver
    counter.Increment()

    // Print the updated count
    fmt.Println("Count:", counter.Count) // Output: 1
}

In this example, the Increment method has a pointer receiver (c *Counter), allowing it to modify the Count field of the Counter instance directly.

{{< notice “note” >}}

  • Methods in Go are functions associated with a particular type.
  • A receiver is a parameter in a method declaration that associates the method with a specific type.
  • Methods can be called on instances of the associated type.
  • Methods with pointer receivers can mutate the state of the receiver, while methods with value receivers cannot.
    {{< /notice >}}

Interfaces in Go

Interfaces in Go provide a way to specify behavior, enabling polymorphism and decoupling of components. Here’s an overview of how to define a basic interface, implement it with a struct and methods, and an example of using interfaces:

Basic Interface:

An interface in Go is defined as a set of method signatures. Any type that implements all the methods of an interface implicitly implements that interface.

package main

import "fmt"

// Define a basic interface named Shape
type Shape interface {
    Area() float64
    Perimeter() float64
}

Struct and Methods:

You can define a struct type and implement the interface methods for that struct.

package main

import (
    "fmt"
    "math"
)

// Define a struct named Circle
type Circle struct {
    Radius float64
}

// Implement the Area method for Circle
func (c Circle) Area() float64 {
    return math.Pi * c.Radius * c.Radius
}

// Implement the Perimeter method for Circle
func (c Circle) Perimeter() float64 {
    return 2 * math.Pi * c.Radius
}

Interface Example:

You can now use the interface and the struct together, treating the struct instances as instances of the interface.

package main

import "fmt"

// Define a basic interface named Shape
type Shape interface {
    Area() float64
    Perimeter() float64
}

// Define a struct named Circle
type Circle struct {
    Radius float64
}

// Implement the Area method for Circle
func (c Circle) Area() float64 {
    return math.Pi * c.Radius * c.Radius
}

// Implement the Perimeter method for Circle
func (c Circle) Perimeter() float64 {
    return 2 * math.Pi * c.Radius
}

func main() {
    // Create a Circle instance
    c := Circle{Radius: 5}

    // Use the interface to call the methods
    var s Shape = c
    fmt.Println("Area:", s.Area())
    fmt.Println("Perimeter:", s.Perimeter())
}

{{< notice “note” >}}
In this example, we define a Shape interface with Area() and Perimeter() methods. We then define a Circle struct with Radius field and implement the Area() and Perimeter() methods for the Circle struct. Finally, we create a Circle instance and use it as an instance of the Shape interface. This demonstrates the power of interfaces in Go, allowing different types to share common behaviors through method signatures.
{{< /notice >}}

Download

Download PDF

Leave a Reply

Your email address will not be published. Required fields are marked *