The Nuances of Constants in Go; Go Isn’t JavaScript
Lane Wagner
Posted on October 30, 2020
The post The Nuances of Constants in Go; Go Isn’t JavaScript first appeared on Qvault.
Constants can be confusing and easy to misuse in Go if you are coming from an untyped language. Let’s take a look at some of the nuanced details of how they work in Go. It’s probably unsurprising, but Go’s constants are almost nothing like JavaScript’s bastardized version of the concept.
Go vs JavaScript
Many programming languages support constants, often denoted by the keyword const
.
Go and JavaScript both declare new constants in the same way:
const frameRate = 60
Constants in Go
- Must be able to be assigned at compile time. The value of a const can’t be the result of a runtime calculation.
- Run faster because the compiler can make specific optimizations.
- Cannot change. The compiler will not allow them to be re-assigned.
- Only work with some types. Arrays, Slices, Maps, Structs, etc… can’t be made constant
- Are not normal Go types unless explicitly assigned as such
Constants in JavaScript
- Can’t be reassigned, but can change. JavaScript’s constants are extremely misleading. the
const
keyword does NOT define a constant value. It defines a constant reference to a value. - If the constant is a type that has inner workings that change, like an array or object then the inner references can be changed.
- Can be assigned using calculated values at runtime, but can’t be re-assigned.
The takeaway if you are coming from JavaScript is that Go’s constants are just different. They deal with compile-time values, not immutable naming.
In Go, constants provide complete safety in regards to the value they hold. They cannot be computed (making them used less often), but are guaranteed to always reference the same value.
In JavaScript, all a const
does is ensure that the same name can’t be changed to reference a different variable in the same scope.
Go’s Constants Must Be Assigned At Compile Time
Constants in Go must be assigned before the program runs. All constants are computed and saved when the program compiles using go build
. Constants can rely on the value of other constants, but not on runtime calculations. For example:
const seconds = 120
const minutes = seconds / 60
Works because both values can be known before the program runs. The following will not work:
func addMinutes(minutes int) {
const more = minutes + 60
return more
}
This won’t work because more
relies on a runtime variable, minutes
. Keep in mind that this would work in JavaScript, because the only rule with constants in javascript is that they can’t be reassigned.
Constants Are Faster
The Go compiler doesn’t need to worry about a constant changing its value, so it can swap every instance of the const with an unchanging number. This makes constants slightly faster.
Should You Use Constants?
Yes. Constants are safer.
Use constants wherever possible. Why would you want to be able to accidentally mutate a value that you know should never change? Let the compiler save you from yourself, and use constants as much as possible.
You may be familiar with the idea that global variables in programming are a bad idea. Variables should typically belong to the smallest scope possible.
Constants in Go don’t apply to the global variable rule, there is nothing wrong with declaring global constants. Granted, if the constant is only used in one place, it may make sense to declare it there. The point however remains: it isn’t dangerous to declare constants globally.
Declare Multiple Constants as a Block
const (
pi = 3.14
timeout = 120 * time.Second
maxGoroutines = 20
)
Only Some Types Can Be Constant
Numeric, boolean, and string types can all be made constant. This includes things like runes, floats, integers, and even custom types that are based on valid underlying types. For example:
type myString string
const lane myString = "wagslane"
Other types like arrays, slices, and maps can not be declared as constant. This makes sense because those types are essentially just pointers, which are addresses of mutable data. However, I have written another article on the elegant ways to get “effectively constant” slices and maps in Go.
By contrast, in JavaScript, anything can be made constant. JavaScript arrays can be declared as constant, but it doesn’t stop the programmer from mutating the elements of the array! The only safety JavaScript’s const
provides is that the variable can’t be explicitly reassigned.
Constants Are Untyped By Default
In Go, variables can have their typed inferred:
thisIsAString := "@wagslane"
Constants, on the other hand, get an untyped flag
const unTypedString = "@wagslane"
An untyped string behaves mostly like a string. That is, its a string type, but doesn’t have a Go value of type string. In order to give it the official Go type of string, it must be declared:
const typedString string = "@wagslane"
Thanks For Reading!
Follow us on Twitter @q_vault if you have any questions or comments
Take some coding courses on our new platform
Subscribe to our Newsletter for more programming articles
Posted on October 30, 2020
Join Our Newsletter. No Spam, Only the good stuff.
Sign up to receive the latest update from our blog.