Branching
Two types ...
if
,else
andelse if
statementsswitch
statement
If ... else syntax
if a%2 == 0 {
// do something
} else if a == 3 {
// do something
} else {
// do something
}
With some inline initialisation ...
if err := validate(something); err != nil {
// do something
}
There are some rules for above initialisation version.
- The initialised variable (
err
above) goes out of scope once theif
block exits. - Only the short hand assignment (
:=
) is allowed.var
keyword is not allowed. - Only some simple assignments are allowed.
Switch statement
Python does not have one. Java has and JavaScript also has.
Go has two types of switch
statements .... expression switch and type switch.
Expression switch
switch <some initialisation>; <switch_expresion> {
case expression1:
// statements. See there are no braces here.
case expression2, expression3:
// statements
default:
// statements
}
- The initialisation is same as
if
. It's scope is limited to theswitch
statement. It is optional. - The
is also optional. If it is not provided and initialisation is provided, then the syntax becomes like switch <some initialisation>; {
. If both are omitted, then the switch expression evaluates totrue
and it looks likeswitch {
. - When the
is given, the value of the is checked against the case expressions for equality. If multiple expressions are given in the same case, then one of them must be equal to the value of for the case to execute. - When the
is not given, then each case expression is evaluated for true
orfalse
. - At least one
case
block (other than thedefault
) is compulsory. - In other languages, the execution falls (goes) through all the subsequent cases after a match is found, unless we use a
break
statement. This is exactly opposite in Go. Here, the when a match is found, only that case's statements are executed and the switch exits. If we want the execution to fall through the subsequent cases as well (like other languages), then we must add afallthrough
statement at the end of the statements of the matching case and all the required subsequent cases. - If no match is found, and if a
default
case is provided, then it will be executed. It acts likeelse
of anif
statement.
Type switch
pass
Looping
Go supports only one type of loop construct ... for
. But it is flexible enough.
It can take three forms broadly.
- Conventional C-type :
for <initialisation>; <condition>; <post_statement> {
- Looping directly over Iterables :
for <condition> {
. This is the only way Python supports. - Looping over unordered collections with the help of
range
keyword.
break
and continue
are available in Go.
Use of range
keyword :
range
iterates over a variety of data structures. The following example from ______ is a very good example.
package main
import "fmt"
func main() {
nums := []int{2, 3, 4}
// `range` on arrays and slices provides both the
// index and value for each entry.
for i, num := range nums {
if num == 3 {
fmt.Println("index:", i)
}
}
// we can ignore any one of the variables with a '_'
for _, num := range nums {
fmt.Println(num)
}
// `range` on map iterates over key/value pairs.
kvs := map[string]string{"a": "apple", "b": "banana"}
for k, v := range kvs {
fmt.Printf("%s -> %s\n", k, v)
}
// `range` can also iterate over just the keys of a map.
// Strange. There is no change on the use of 'range' keyword on the
// right hand side of the assignment. It's only that we are
// now assigning to one variable on the left side.
for k := range kvs {
fmt.Println("key:", k)
}
// `range` on strings iterates over Unicode code
// points. The first value is the starting byte index
// of the `rune` and the second the `rune` itself.
// This very important. Conventional c-type looping over the length of
// a string, gives the individual bytes, not the runes.
for i, c := range "go" {
fmt.Println(i, c)
}
}
range
on slices and arrays works somewhat like theenumerate()
function (from 'collections' module) in Python.- Just like in Python, we can ignore some return values (in assignments) with underscore
_
. range
on maps is strange. If we assign to two variables, then we get both the key value pair. If we assign to one variable only, then we get only the key. I don't understand why the second way is made available when we can ignore any of the returned variables (key or value) with the underscore_
assignment. Likefor key, _ := range some_map {
. But, it seems Go does not recommend that way. In fact, thegofmt
tool automatically converts such statements to single assignments.range
on strings also is noteworthy. Given that a string is just a sequence of bytes,range
does not just simply iterates over the bytes. Rather, it returns unicode code points (orrune
type values). This is always the preferred way for looping over the characters of a string, in that case.