Comprehensive Guide to Dates and Times in Go

Keeping track of time in code has long been a developer’s nightmare. While no language or package manages time perfectly, I’m of the opinion that Golang does a pretty good job out-of-the-box. This full tutorial is designed to answer ~90% of the questions you’ll have about managing time in Go.

Sorry to interrupt! I just wanted to mention that you should check out my new free Go course. It’s designed to teach you all the fundamentals of my favorite coding language.

Table of Contents

Overview – How dates and times are stored in Go

The first thing to know is that you probably don’t need any third-party packages to manage times and dates in Go. The Go standard library’s time package is very robust and can do almost anything you’re going to want to do.

The default time.Time type represents an instant in time with nanosecond precision. It’s a struct that has no exported fields, which means you’ll never need to use a dot operator to access different fields. Instead, various methods are available to get the data you need. For example, time.Year() returns the year (as an integer) of the time object in question.

The two most common ways to create a new time object are to use the current time, or to provide a date as input.

Get the current time with time.Now()

The time.Now() function returns the current local time. If you work on the backend, it’s likely you’ll also want to immediately convert it to UTC.

currentTime := time.Now() currentTimeUTC := time.Now().UTC()
Code language: Go (go)

Create a time object for a specific date

If instead you want a time object for a specific date, you can use the time.Date() function.

// takes a year, month, day, hour, minute, second, nanosecond, and location someonesBirthday := time.Date(1990, time.May, 10, 23, 12, 5, 3, time.UTC)
Code language: Go (go)

Printing, parsing, and formatting times in Go

While dates and times are typically stored as time.Time objects within Go programs, we frequently need to save them to databases, marshal them to JSON, or just print them to the console. It’s nice that Go provides functions to format and parse dates easily. That said, the way it’s handled is unique compared to most coding languages.

Let’s say we have a time object and we want to be able to print in a specific format. The Go formatting engine takes a layout of specific constants and uses that as an example for how to format the time.

The reference time is defined in the docs as:

Mon Jan 2 15:04:05 -0700 MST 2006
Code language: CSS (css)

So if we want to format our time object a specific way, we can just use the constants from the reference time but rearrange them how we want.

t := time.Now().UTC() fmt.Println(t.Format("2006 01 02 MST")) // prints 2021 05 16 UTC (assuming that's the current time)
Code language: Go (go)

The time.Parse() function works the same way, but takes a time string and a layout as an input, and attempts to create a new time object.

t, err := time.Parse("2006 01 02 MST", "2021 05 16 UTC") if err != nil{ log.Fatal(err) } fmt.Println(t) // Prints "2021-05-16 00:00:00 +0000 UTC" assuming that's the current time // The above is the default printing format for a time object (rfc3339)
Code language: Go (go)

As I mentioned above in the comment, the default parsing and formatting layout for Go is rfc3339. Where possible, if your team works primarily in Go, I’d recommend using the default formatting,, it’s the default for a reason.

Time durations

My bane in programming is when developers don’t include units in their calculations. Inevitably one developer assumes the variable timeElapsed (an int) represents seconds, it really represents milliseconds. In Go, this isn’t a problem as long as everyone adheres to the standard of the time.Duration type.

Durations are just a specific kind of int64. They represent the elapsed time between two instants as a nanosecond count. the only drawback is that the largest representable duration is ~290 years, which hasn’t been a problem for me yet. There are several constants defined by the time package to represent some common durations.

fiveSeconds := time.Second * 5 sixMinutes := time.Minute * 6 oneDay := time.Hour * 24
Code language: Go (go)

Convert between separate timezones and locations

Every time.Time object is associated with a location, which is basically a timezone. 5 o’clock is meaningless if you don’t know which timezone it’s in. Locations are defined by the time.Location type, which, like the time.Time type, is a struct with no exported fields.

Get the timezone from an existing time object

myTime := time.Now() myTimezone := myTime.Location()
Code language: Go (go)

Create a new time.Location object

<code>mstLocation, err := time.LoadLocation("MST")</code>
Code language: Go (go)

Convert a time from one location to another

<code>loc, err = time.LoadLocation("MST")</code>if err != nil{ log.Fatal(err) } mstTime := t.In(loc)
Code language: Go (go)

Custom timezone name

A timezone is basically just a name and a duration offset from UTC. If you want a specific timezone but want to change its name you can do that.

tzName := "CUSTOM-TZ" tzOffset :=60*60*5 // seconds east of UTC loc := time.FixedZone(tzName, tzOffset)
Code language: Go (go)

Add, subtract and compare times

Times and durations naturally work well together, and several helper functions are available to do basic time arithmetic and comparisons.

Add time and duration

There are two primary functions for adding time to an existing time. Keep in mind, these functions also work for subtracting time, you just add a negative duration.

time.Add()

myTime := time.Now().UTC() inTenMinutes := myTime.Add(time.Minute * 10) // inTenMinutes is 10 minutes in the future myTime = time.Now().UTC() tenMinutesAgo := myTime.Add(-time.Minute * 10) // tenMinutesAgo is 10 minutes in the past
Code language: Go (go)

time.AddDate()

myTime := time.Now().UTC() // adds years, months, and days inOneMonth := myTime.AddDate(0, 1, 0) // inOneMonth is 1 month in the future myTime = time.Now().UTC() twoDaysAgo := myTime.AddDate(0, 0, -2) // twoDaysAgo is 2 days in the past
Code language: Go (go)

Get difference between two times

The sub() function gets the difference between two times. Keep in mind, the sub() function does not subtract a duration from a time. You, perhaps counterintuitively, need to use the add() function for that.

start := time.Date(2020, 2, 1, 3, 0, 0, 0, time.UTC) end := time.Date(2021, 2, 1, 12, 0, 0, 0, time.UTC) <code>difference := end.Sub(start) fmt.Printf("difference = %v\n", difference)</code>
Code language: Go (go)

Compare two times to see which comes after the other


There are two functions that should take care of most of your time comparison needs.

time.After()

first := time.Date(2020, 2, 1, 3, 0, 0, 0, time.UTC) second := time.Date(2021, 2, 1, 12, 0, 0, 0, time.UTC) isFirstAfter := first.After(second)
Code language: Go (go)

time.Equal()

first := time.Date(2020, 2, 1, 3, 0, 0, 0, time.UTC) second := time.Date(2021, 2, 1, 12, 0, 0, 0, time.UTC) equal := first.Equal(second) // equal is true if the both times refer to the same instant // two times are equal even if they are in different locations
Code language: Go (go)

Intervals, sleeping, and tickers

If you need your program to synchronously sleep, there’s an easy way to do that, time.Sleep(). Keep in mind, this is synchronous, it will block the current goroutine.

Force the current goroutine to sleep

fmt.Println("hello") time.Sleep(time.Second * 2) fmt.Println("world 2 seconds in the future")
Code language: Go (go)

Execute code on an interval using tickers

If you need to do something an a fixed interval, the time.Ticker type makes it easy to do so.

func <strong>doSomethingWithRateLimit</strong>() { ticker := time.<strong>NewTicker</strong>(time.Second) for range ticker.C { // doSomething() is executed every second forever <strong>doSomething</strong>() } }
Code language: Go (go)

The first tick to come through the ticker channel is after the first duration. If you want an immediate first tick you can read about that here.

Saving memory with TinyDate and TinyTime

A typical time.Time value takes ~24 bytes in memory. Sometimes, you can’t afford to use that much. If you’re in that situation then check out my TinyTime and TinyDate libraries on GitHub. They only use 4 bytes each!

Have questions or feedback?

Follow and hit me up on Twitter @q_vault if you have any questions or comments. If I’ve made a mistake in the article, please let me know so I can get it corrected!