Golang: Sort Package Introduction

Edit

Published on January 15, 2024

Author: Meet Rajesh Gor

#go

I have been learning Golang for around 2 years now, and I have never paid attention to the sort package, can you believe that! Where was this package hiding?

The sort package provides convenient methods for sorting slices and user-defined collections in Go. Here's a quick overview of what it can do:

Sorting Slices

To sort a slice of builtin types like ints, strings, etc, you can simply call sort.Slice():


go
nums := []int{5, 2, 6, 3, 1}

sort.Slice(nums, func(i, j int) bool {
    return nums[i] < nums[j]
})

// or 
// sorts.Ints(nums)

fmt.Println(nums)

bash
$ go run main.go

[1, 2, 3, 5, 6]

The sort is in-place, mutating the original slice. You can customize the sort order by providing a less(i, j int) bool function.

Sorting Struct Slices

To sort a slice of custom structs, add a Len(), Less(i, j int) bool, and Swap(i, j int) method:


go
type Person struct {
  Name string
  Age  int
}

people := []Person{
  {"Bob", 30},
  {"John", 20},
  {"Alice", 25},
}

sort.Slice(people, func(i, j int) bool {
  return people[i].Age < people[j].Age
}) 

fmt.Println(person)

bash
$ go run main.go
[{"Name":"Alice","Age":25},{"Name":"Bob","Age":30},{"Name":"John","Age":20}]

This will sort people by age.

Sorting Maps

Maps are inherently unordered in Go. We can't sort them, but we can iterate them in a sorted way. We need a separate slice of keys or values(whicher required), we will sort those keys/values and iterate over them in that order.:

Sort by Keys

To sort a map by keys, we can use the sort.Strings() function or any other sort function as per the data structure, there are functions like sort.Ints, sort.Float64, or sort.Slice. We can create a new slice of keys from the map and then apply the sort on that newly created slice. After the slice of keys is created, we can iterate over it and access the map values in a order of sorted keys.


go
counts := map[string]int{
  "hello": 5,
  "world": 2,
  "foo": 3,
}

keys := make([]string, 0, len(counts))
// extract keys 
for k := range counts {
  keys = append(keys, k)
} 

// sort keys
sort.Strings(keys) 

// iterate with the sorted keys slice
for _, k := range keys {
  fmt.Println(k, counts[k]) 
}

bash
$ go run main.go

foo 3
hello 5
world 2

This prints the map ordered by key. You can sort by value too.

Sort by Value

To iterate the map with sorting order of values, we can approach it in a similar way. We create a slice of keys. This time, we don't sort the keys, instead, we change the position of the slice of keys depending on the values. This changes the key order based on the sorted values in the map. By similarly iterating over the key slice, we iterate over the map in a sorted order of the values.


go
counts := map[string]int{
  "hello": 5,
  "world": 2,
  "foo": 3,
}

keys := make([]string, 0, len(counts))
// extract keys
for k := range counts {
  keys = append(keys, k)
}

// sort by value
// i.e. change the order of key based on values sort order
sort.SliceStable(keys, func(i, j int) bool {
    return counts[keys[i]] < counts[keys[j]]
})

// iterate sorted
for _, k := range keys {
  fmt.Println(k, counts[k])
}

bash
$ go run main.go

world 2
foo 3
hello 5

The sort.SliceStable function is used to sort the slice in place. It takes in a slice and the less function which is the actual logic for comparison in the values. This function sorts the key elements based on the values in the map, and thereby the key slice is shuffled by the sorted order of the values in the map.

The sort package is super useful for quickly organizing Go data structures. There are Find which tries to return a index of the first element that satisfies a comparison condition with a flag or found or not with a boolean. There is also a Search which is used specifically for searching elements in a sorted array, it also gives a first index of the occuring element. But that is a topic for another day!

Thank you, Happy Coding :)

<a class='prev' href='/django-form-load-frontmatter'>

    <svg width="50px" height="50px" viewbox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
        <path d="M13.5 8.25L9.75 12L13.5 15.75" stroke="var(--prevnext-color-angle)" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"> </path>
    </svg>
    <div class='prevnext-text'>
        <p class='prevnext-subtitle'>prev</p>
        <p class='prevnext-title'>Django Blog DevLog: Load Frontmatter data into Template/Model Form Fields</p>
    </div>
</a>

<a class='next' href='/django-bulk-update-queryset'>

    <div class='prevnext-text'>
        <p class='prevnext-subtitle'>next</p>
        <p class='prevnext-title'>Django Bulk Update QuerySet objects</p>
    </div>
    <svg width="50px" height="50px" viewbox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
        <path d="M10.5 15.75L14.25 12L10.5 8.25" stroke="var(--prevnext-color-angle)" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"></path>
    </svg>
</a>