DEV Community

Lane Wagner
Lane Wagner

Posted on • Originally published at qvault.io on

The Nuances of Constants in Go; Go Isn’t JavaScript

The Nuances of Constants in Go; Go Isn't JavaScript

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
Enter fullscreen mode Exit fullscreen mode

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
Enter fullscreen mode Exit fullscreen mode

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
}
Enter fullscreen mode Exit fullscreen mode

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
)
Enter fullscreen mode Exit fullscreen mode

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"
Enter fullscreen mode Exit fullscreen mode

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"
Enter fullscreen mode Exit fullscreen mode

Constants, on the other hand, get an untyped flag

const unTypedString = "@wagslane"
Enter fullscreen mode Exit fullscreen mode

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"
Enter fullscreen mode Exit fullscreen mode

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

Top comments (0)