Slices and Arrays in Go¶
(The following is based largely on http://blog.golang.org/slices.)
Like most programming languages, Go has a data structure called an array that stores a sequence of objects of the same type. Less commonly, Go has a second — and more useful — data structure called a slice that lets you refer to a contiguous sub-sequence of elements from an underlying array.
Arrays¶
Arrays are a fundamental data structure in Go. For instance:
// declares arr to be an array of 5 ints, all initialized to 0
var arr [5]int
for i, val := range arr { // print arr
fmt.Printf("arr[%d] = %d\n", i, val)
}
This prints:
arr[0] = 0
arr[1] = 0
arr[2] = 0
arr[3] = 0
arr[4] = 0
When you create arrays, Go always initializes its element to the zero-value for the array type.
You can also create an array using an array literal, e.g.:
var arr = [5]int{1, 2, 3, 4, 5} // array literal
Ranged for-loops (i.e. loops that use the keyword range
as above) are
often the nicest way to traverse arrays in Go. You can also use a more
traditional C-style loop:
for i := 0; i < len(arr); i++ {
fmt.Printf("arr[%d] = %d\n", i, arr[i])
}
The len
function returns the length of an array, while the usual square-
bracket notion is used to access individual elements at a given position. The
first index position of a Go array is always 0.
You can’t change the size of an array in Go. Once an array is created, it’s length remains fixed until it is disposed of by the garbage collector.
You can pass an array to a function. As with all parameters, Go passes arrays by value (i.e. it makes a copy of the array):
func add1(arr [5]int) {
for i := range arr {
arr[i]++
}
}
Now call it with this code:
var arr [5]int // arr is initialized to all 0s
add1(arr) // add 1 to every element of arr
for i, val := range arr { // print the results
fmt.Printf("arr[%d] = %d\n", i, val)
}
This prints:
arr[0] = 0
arr[1] = 0
arr[2] = 0
arr[3] = 0
arr[4] = 0
Nothing has changed because add1
modified the copy of arr
, not the
original.
Copying an array is generally inefficient, and one way around this is to re-
write add1
to accept a pointer to the array, e.g.:
func add1(arr *[5]int) {
for i := range *arr {
(*arr)[i]++
}
}
Then you call it like this:
var arr [5]int
add1(&arr) // note the &
for i, val := range arr {
fmt.Printf("arr[%d] = %d\n", i, val)
}
Now the original arr
is changed:
arr[0] = 1
arr[1] = 1
arr[2] = 1
arr[3] = 1
arr[4] = 1
The code for this is rather messy. It’s easy to forget a *
or &
.
Arrays have a significant limitation. Consider this slight change:
var arr [6]int // 6 elements instead of 5
add1(&arr)
for i, val := range arr {
fmt.Printf("arr[%d] = %d\n", i, val)
}
This doesn’t even compile! You get a type mis-match error because add1
expects an array of length 5, but we are passing it an array of length 6. In
Go, the length of an array is part of the array’s type, so arrays of
different lengths have different types.
We get the same error in the version without pointers:
func add1(arr [5]int) {
for i := range arr {
arr[i]++
}
}
func main() {
var arr [6]int // 6 elements instead of 5
add1(arr)
for i, val := range arr {
fmt.Printf("arr[%d] = %d\n", i, val)
}
}
Again, this is a type mis-match error: you can’t pass a 6-element array to a function expecting a 5-element array because an array’s length is part of its type.
This is a big problem with arrays. It essentially makes it impossible for us to write one function that works on arrays of different sizes.
Go addresses these problems with arrays by introducing a new data structure called a slice.
Slicing an Array¶
A slice is a data structure that describes a contiguous sequence of elements in an array. A slice is not an array, but for most purposes it acts like one.
One way to make a slice is from an array:
var arr [5]int // an array with 5 elements
var sl []int = arr[1:4] // a slice referring to arr[1], arr[2], arr[3]
Here, sl
is a slice, and it’s type is []int
, read “slice of ints”. We
can also define sl
like this:
var sl = arr[1:4] // type of sl inferred from arr[1:4]
Or this:
sl := arr[1:4] // can only use := if this line is in a function
You can use a slice pretty much the same way that you use an array:
var arr [5]int
for i := range arr { // arr is {0, 1, 2, 3, 4}
arr[i] = i
}
sl := arr[1:4] // sl is a slice of length 3
for i := 0; i < len(sl); i++ { // print sl
fmt.Printf("sl[%d] = %d\n", i, sl[i])
}
It’s important to understand that a slice always refers to an underlying array, and if that array changes, then so to does the slice:
arr[1] = -100 // change second element of arr
for i := 0; i < len(sl); i++ { // print sl
fmt.Printf("sl[%d] = %d\n", i, sl[i])
}
This prints:
sl[0] = -100
sl[1] = 2
sl[2] = 3
You can pass slices to functions like this:
// sl can be any int slice of any length
func display(sl []int) {
for i, val := range sl {
fmt.Printf("sl[%d] = %d\n", i, val)
}
}
func main() {
var arr [5]int
sl1 := arr[1:4] // sl1 refers to arr[1], arr[2], arr[3]
display(sl1)
sl2 := arr[2:5] // sl2 refers to arr[2], arr[3], arr[4]
display(sl2)
}
Unlike arrays, the length of a slice is not part of its type, so the
display
function can be passed any int
slice.
Slice Notation in General¶
Suppose that arr
is of length n
, and consider this statement:
s := arr[a:b]
Since the indices of the underlying array arr
start at 0 and end at n
- 1, a few basic restrictions on a
and b
are clear:
Both
a
andb
must be non-negative. This is because the 0 is the first index for the underlying array (all Go arrays start at 0).You will get either a compiler error or a run-time error if you use negative indices. For example, this causes a compiler error:
var arr [5]int s := arr[-2:4] // compiler error: negative index not allowed in a slice
Or sometimes you get a run-time error like this:
var arr [5]int i := -2 s := arr[i:4] // runtime error when this line is executed fmt.Println(s[0]) // s must be used somewhere to avoid an // "s is unused error"
It’s unfortunate that Go can’t recognize this indexing error at compile time. But, in general, the compiler can’t always know the value of a variable before the program runs. For example, if
i
was read from the user, then the value ofi
is not known until the user types it.Both
a
andb
must be less than, or equal to,n
. Again, this is a consequence of the fact that the underlying array hasn
elements.An important detail is that a slice may refer to location
n
, even througharr[n]
does not exist. For example:var arr [5]int s := arr[0:5] // okay: s is the entire array s = arr[0:4] // okay: s is the first 4 elements of the array s = arr[5:5] // okay: s is the empty slice, []
Note that we use
:=
only for the first assignment tos
. That’s because:=
assigns and creates the variable, and it would be an error to creates
more than once.The first index,
a
, must be less than, or equal to,b
. Otherwise, you get an inverted slice error, e.g.:var arr [5]int s := arr[3:1] // compiler error: slice indices are inverted
To summarize, if arr
is an array of non-negative length n
, then the
slice arr[a:b]
is a valid slice if 0 <= a
<= b
<= n
.
Finally, there are a couple of slice indexing shorthands that are useful to
know. Assuming arr
is of length n
, then:
arr[a:]
is the same asarr[a:n]
arr[:b]
is the same asarr[0:b]
arr[:]
is the same asarr[a:n]
Slices of Slices¶
We’ve seen that you can create a slice from an array. You can also create a slice from another slice. For example:
var arr [5]int
for i := range arr { arr[i] = i }
s1 := arr[1:4] // a slice of an array
s2 := s1[0:2] // a slice of a slice
fmt.Printf("s1 is %v\n", s1) // %v prints the value of a Go object
fmt.Printf("s2 is %v\n", s2) // in a standard notation
This prints:
s1 is [1 2 3]
s2 is [1 2]
Both slices refer to the same underlying array, so if that changes, the slices change too:
// ... code same as above ...
arr[1] = 77
fmt.Printf("\ns1 is %v\n", s1)
fmt.Printf("s2 is %v\n", s2)
The entire program prints:
s1 is [1 2 3]
s2 is [1 2]
s1 is [77 2 3]
s2 is [77 2]
We can also change the underlying array through a slice, e.g.:
// ... code same as above two examples ...
s2[0] = -55
fmt.Printf("\narr is %v\n", arr)
fmt.Printf("s1 is %v\n", s1)
fmt.Printf("s2 is %v\n", s2)
The entire program now prints:
s1 is [1 2 3]
s2 is [1 2]
s1 is [77 2 3]
s2 is [77 2]
arr is [0 -55 2 3 4]
s1 is [-55 2 3]
s2 is [-55 2]
How Slices are Implemented¶
It’s instructive to see how slices are implemented in Go, as that can help explain their behaviour. Conceptually, a slice could be implemented like this:
type sliceHeader struct {
Length int
Capacity int
ZerothElement *int // pointer to an int
}
Note: We can’t mimic slices perfectly because every variable in a Go
structure must have an explicit type. Here, we’ve given ZerothElement
the
type *int
, so sliceHeader
can only refer to arrays of int
s. In
general, the ZerothElement
must be of type T*
, where T
is the type
of the elements in the underlying array. Go currently has no way of
expressing T
: that would require something like templates as in C++ or
Java.
Importantly, a sliceHeader
uses a small, fixed amount of memory. It
consists of two integers and a pointer. The elements of the slice are stored
in the underlying array. Thus, the cost of creating or copying a slice is
extremely low.
The capacity of a slice is the number of elements in the underlying array from
the start of the slice to the end of the array (not the end of the slice). The
function cap
returns a slice’s capacity:
var arr [5]int
s := arr[1:4] // a slice of an array
fmt.Printf("length of s is %d\n", len(s))
fmt.Printf("capacity of s is %d\n", cap(s))
This prints:
length of s is 3
capacity of s is 4
So why does a slice store the capacity? The capacity of a slice is used for extending its size. Basically, you can extend the size of a slice up to its capacity. If you extend it past its capacity, you get an error.
Consider this function for appending an integer onto the end of a slice:
func extend(slice []int, x int) []int {
n := len(slice)
slice = slice[0 : n+1] // can only slice up to a slice's capacity
slice[n] = x
return slice
}
Then run this code:
func main() {
var arr [5]int
s := arr[0:0] // initially empty slice: length 0, capacity 5
s = extend(s, 1)
s = extend(s, 2)
s = extend(s, 3)
s = extend(s, 4)
s = extend(s, 5)
fmt.Printf("s is %v\n", s)
s = extend(s, 6) // runtime error: slice bounds out of range
}
The sixth time extend
is called gives a runtime error because there is no
more space in the slice’s underlying array. A slice can never be longer than
the array it refers to.
Passing a Slice to a Function¶
When you pass an array to a function, the array is passed by value, i.e. a copy of the array is made. This is both inefficient for large arrays, and also a limitation because the function cannot modify the original array (it can only modify the copy).
Slices behave differently. When you pass a slice to a function, the slice data structure is copied, but the underlying array is not copied. The copied slice points to the same underlying array as the original slice, and so both slices access the same array. Thus passing a slice to a function is not only efficient, but it lets you modify the underlying array.
For example:
func add1(s []int) { // s is slice
for i := range s {
s[i]++
}
}
func main() {
var arr [5]int // arr is an array
s := arr[0:5]
add1(s)
for i, val := range s {
fmt.Printf("s[%d] = %d\n", i, val)
}
}
This prints:
s[0] = 1
s[1] = 1
s[2] = 1
s[3] = 1
s[4] = 1
Making Slices with make¶
We’ve seen that one way to create a slice is directly from an array. Another
option is to use Go’s make
function like this:
s := make([]int, 10, 15) // makes an int slice of length 10, capacity 15
This statement creates both an array of ints of length 15, and a slice of length 10 (and capacity 15) that refers to that array.
If you want to create a slice whose length and capacity are equal, you don’t
need to pass the third parameter to make
:
s := make([]int, 10) // same as make([]int, 10, 10)
If you need to increase the length of a slice past its capacity, your only
choice is to create a new slice. This is usually done using the append
function discussed below, but here we will do it “by hand” using make
:
s := make([]int, 10, 15)
fmt.Printf("length of s is %d\n", len(s))
fmt.Printf("capacity of s is %d\n", cap(s))
// make a new slice with the same length of s but twice the capacity
s2 := make([]int, len(s), 2 * cap(s))
for i := range s { // copy values of s into s2
s2[i] = s[i]
}
s = s2
fmt.Printf("\nlength of s is %d\n", len(s))
fmt.Printf("capacity of s is %d\n", cap(s))
The copy Function¶
In the example code above we copied the values from one slice to another using
a loop. Copying slices is common enough in Go that a built-in copying
function is provided. Conveniently, it’s called copy
:
s1 := []int{1, 2, 3} // slice literal
s2 := make([]int, len(s1), 2 * cap(s1))
copy(s2, s1) // copy values from s1 into s2
fmt.Printf("%v\n", s2) // prints: [1 2 3]
Notice that s1
is initialized using a slice literal, which is yet another
way to create a slice. If we wrote [3]int{1, 2, 3}
, then it would instead
be an array literal.
Not only is copy
easier to type than a loop, but it correctly handles a
few special cases. First, the number of values copied by copy
equals the
minimum of the lengths of its two arguments. For example:
s1 := []int{1, 2, 3} // slice literal
s2 := make([]int, 2) // slice of length 2, capacity 2
copy(s2, s1) // copy values from s1 into s2
fmt.Printf("s2 = %v\n", s2) // prints: [1 2]
The other thing copy
does is to correctly handle over-lapping slices
within the same underlying array. This lets you use copy
to move items
around inside a slice. For example:
s := []int{0, 1, 2, 3, 4} // s is a slice of ints
copy(s[3:5], s[2:4]) // move [2 3] one position up in s
fmt.Printf("s = %v\n", s) // prints: [0 1 2 2 3]
The append Function¶
What happens if you want a slice to grow longer than it’s capacity? As
mentioned above, once a slice is created it’s capacity can’t change because
it’s limited by the length of the underlying array (which can’t change). So
you need to create a new, longer slice, and this is what the built-in
append
function does (when necessary).
For example:
s := []int{0, 1, 2, 3, 4}
fmt.Printf("len(s) is %d\n", len(s))
fmt.Printf("cap(s) is %d\n", cap(s))
s = append(s, 5) // append one new element onto the end of s
fmt.Printf("\ns = %v\n", s)
fmt.Printf("len(s) is %d\n", len(s))
fmt.Printf("cap(s) is %d\n", cap(s))
Running this code prints this:
len(s) is 5
cap(s) is 5
s = [0 1 2 3 4 5]
len(s) is 6
cap(s) is 12
Appending 1 new element to s
increases its length by 1, but its capacity
has increased from 5 to 12. That’s because when append
is forced to
increase the slice’s capacity, it allocates more memory than is strictly
needed so that future appends will be faster. Note that the documentation for
append does not say exactly how
much the capacity of the array will be increased, and so it is possible that
different versions of Go might print different capacities in this example.
The implementation of append
is something like this:
func Append(s []int, elements ...int) []int {
n := len(s)
total := len(s) + len(elements)
if total > cap(s) {
// reallocate
newSize := 2 * total + 1
newSlice := make([]int, total, newSize)
copy(newSlice, s)
s = newSlice
}
s = s[:total]
copy(s[n:], elements)
return s
}
(This only works with int
slices, while the built-in append
works with
any type of slice. But the essential code is the same.)
As you can see, a new slice is created only when there is no more capacity in
s
to hold all everything in elements
. Thus appending is often
extremely efficient, and a new slice is only created when necessary.
One new feature used here is the argument type ...int
. This lets you pass
in a single int
, or an entire slice of ints. For example:
s := []int{0, 1, 2, 3, 4}
t := []int{5, 6, 7}
s = append(s, t...) // the ... is necessary
fmt.Printf("\ns = %v\n", s)
fmt.Printf("len(s) is %d\n", len(s))
fmt.Printf("cap(s) is %d\n", cap(s))
Exercises¶
- Write a function called
shrink(s []int)
that returns a new slice with the same elements ass
, except the capacity of the newly returned slice is the same as its length.