A set of small programs that each performs a specific task, which demonstrates certain feature(s) of Go.
Riyan is responsible for all the Go code, but CS Professor Ian Harris of UCI supplies the inspiration.
- 0. Requisites
- Data Types
- Formats
- Functions
- Object Orientation (in Go)
- 8. Farm Animals
- 9. Farm Animals 2 (has Interface and Reflection :)
- Concurrency
To test these programs, please ensure you have Go installed on your machine.
You can do that by navigating to your Terminal and running this command:
go version
git version
If they return without error, you have them installed and you can proceed with an empty folder, and executing this line in that empty folder:
git pull https://github.com/ryechris/Go-Cookbook.git
And, of course, you can run the files with this format:
go run <filename>.go
go run 01-truncate-decimals.go
This program prompts the user to type a floating number (for example, 3.1415). It then removes the decimal point and everything to the right of it (e.g. input: 3.14; result: 3).
Though small and clean, it demonstrates clearly the following concepts:
- User input
- Type handling & conversion
go run 02-find-ibm.go
The above line prompts the user to enter a string. Any string.
The program then searches that string to see if the characters "i", "b", or "m" are present -- in that order.
For example, "icbm" returns Found, but "mbi" does not.
The letter-finding program succeeds regardles of cases (uppercase, lowercase, or a mix) in the string.
This program plays with:
- Arrays/Slices
- Handling type string
- Using the strings package
go run 03-sort-numbers.go
Go's array has a fixed length: it can only contain as many elements as what you set it to be when you create it. However, in this program, you can add as many numbers as you want to the container. The way we do that is with a slice, which is like a window to an array, instead of an array itself. That way, we can keep increasing the size of the slice to contain every number you place in it.
So the concepts we cover in this program are:
- Slices (of varying length)
- Packages
go run 04-make-json.go
A popular standard, JSON facilitates information exchange. Programs often need to communicate with other systems. Since systems vary, standards are important.
Go just happens to provide a json package: its standardized implementation to encode JSON from its various structures.
This program demonstrates Go's ability to communicate in JSON. It prompts the user to enter a name, and then an address. It maps the address to the name, and the program returns the data's JSON object.
Go:
map[Name:Address]
JSON:
{"Name":"Address"}
Hence, this program contains the following concepts:
- Map
- JSON
Programs of course must be able to talk with files. So here we illustrate Go's ability to do that.
When you did git pull, included in the folder are two example files:
05-file1.txt
05-file2.txt
Each of these files has a list of names: on each line are two name values, separated by a comma.
George,Washington
Abraham,Lincoln
We will use these files in the program. When the user runs the following line:
go run 05-read-file.go
the program presents two options: file1 or file2. If the user types 1, the program processes file 1. If the user types 2, the program processes file 2. If the user types something else, the program will output an error.
The program processes the file by
- opening the file
- reading the file
- closing the file In closing, it returns the data to the user in this format:
First Name: George
Last Name: Washington
First Name: Abraham
Last Name: Washington
Of course we can do anything with that data, such as printing it in JSON. But we have already done a JSON example in file 04, so here we show another capability of Go programs.
In so doing, we are able to focus more on fewer ideas -- since it is a cookbook -- instead of having too many ingredients in the program:
- File Access
- Slice: type, iteration
- Flow control: loop, if, switch
- Functions
go run 06-sort-bubble.go
We continue our theme of code organization from program 05.
However, program 06 is not a continuation of prog 05.
This program asks input from user, and places them into an array (slice, actually). The program then sorts the numbers in ascending order. When it concludes, it displays the result to the user.
For example, in organizing the code, lines that repeat often is placed into its own function for a higher-quality code writing.
Note: The algorithm used here, the Bubble Sort algorithm, is a poor choice for production. Its inefficiency is pointed out by the comments in the code. But, the algorithm is useful to demonstrate the following topics:
- Organization
- Control Flow: nested loops
- Functions
Having covered functions in the previous recipe, we are now going to have fun to dive deeper into functions!
We assume the user of this .go file
go run 07-compute-displacement.go
has some observed an object moving with a constant acceleration, such as a pen falling from a table. Hence, the program requests three pieces of data:
- the constant acceleration (e.g. gravity, 9.8 m/s2)
- the velocity of the object at the start of the observation (e.g. object starts at rest, 0 m/s)
- and the starting displacement (e.g. 0 m)
With these values reported (inputted) by the user, we will compute displacement.
In the process, we learn about functions in Go being First Class values:
- Functions as variables
- Functions as return values
- Function's environment & lexical scoping
This recipe is about how to do Object Oriented Programming in Go.
Go does not have a class the way other Object Oriented languages have, such as Python.
However, Go has something equivalent to class. To illustrate object oriented programming, a farm is locus classicus. It is "real life": it has objects (the animals), and the animals has methods (they have verbs: they can eat, walk, etc.).
Hence, animal farm is more than a classic example: it is an excellent way to illustrate how we do OOP in Go.
In Go, we "create a class" by associating data with methods.
If there are various data types -- like a class often has -- you can place them in a struct:
type Point struct {
x <some Type>
y <any Type>
...
}
After the struct, you make methods. And, you ensure those methods are available only for the struct you have in mind.
Topics:
- Object Oriented Programming
- Go's class equivalence: struct/data with methods.
- Receivers
This time it gets more exciting. The Farm Animals continue. It continues because we're not done with Object Oriented Programming. We are not done with OOP because we have not covered something that is always discussed in OOP: polymorphism.
Polymorphism is when a thing is able to take on multiple forms. Two constructs that enable polymorphism is inheritance and overriding. For example, looking at our animal farm, suppose the object-oriented programmer creates a class for animal. It has a method Speak(), because animals can produce sounds. But these sounds vary: a cat meows, and a dog woofs. So what does the object programmer do? He creates a subclass for each animal: Cat for cats, Dog for dogs.
A subclass is called so because it inherits the data and methods from the class, and because it can override them. For example, the Cat subclass inherits from the Animal class. But it overrides the Speak() method, because cats meow: they don't moo, they don't woof or any other noise. In this case, we say that the Speak() method is polymorphic, because although the method (inc method signature) is the same, it may differ from subclass to subclass. In the afore example, Speak() in Cat is different from Speak() in Dog, as you would observe in "real life."
So evidently this is an important feature. But Go does not offer class or subclass or inheritance or overriding. So how do we do polymorphism in Go?
This recipe is all about how we do polymorphism in Go. It demonstrates that Go's interface accomplishes both inheritance and overriding, but without those concepts of inheritance & overriding.
In the first Animal Farm, Animal is a struct. Here, Animal is an interface. And we create structs for the "objects" (e.g. dog, pig) and making them satisfy the interface. We show how that is done in Go: there is no explicit statement that they satisfy the interface. They just do, and we know they do as long as they implement all the methods in the interface.
Also, in the first Animal Farm, there is no option to create a new animal. Now in Animal Farm 2, the user can make one new animal. To accomplish this feature, we have to modify the struct for the new animal -- at runtime. To do that, we call upon the powerful Reflection, and we get to see it in action.
Topics:
- Polymorphism: how we accomplish that with Go
- Interface: what it is and an example
- Reflection: an example of it in action
This multi-threaded application computes factorials. When you run
go run 10-compute-factorial.go
You input the number you want the factorial of. For example, if you want 5!, you enter 5. The program will then compute 5! = 5 * 4 * 3 * 2 * 1 = 120. Due to the size of integers, 20 is the largest integer it accepts. (Because 20! is already large at 20! = 2,432,902,008,176,640,000)
Computing factorial is relatively compute-intensive. Instead of doing it in one sequence, we use a total of 7 goroutines (including the Main one) to obtain the factorial. The program divides the list of integers (e.g. 8!: 8 * 7 * 6... * 1) into four, then it assigns a goroutine to compute the factorial of each (e.g. 8 * 7, 6 5.... 21) Then, it takes the resultant numbers and repeat the process, with one difference: we divide the list into two, and use two goroutines to handle it. The final result is two numbers, and the main goroutine. (see code for the actual sequence of lines)
So this is concurrent programming, with many threads running concurrently. Concurrency really happens at the machine-code instructions, we have no idea of the interleavings that can or do take place. Since we have no visibility into the interleavings, many things can go wrong.
But nothing goes wrong in this program. Nothing goes wrong here because we ensure excellent communication among the threads (the goroutines) by following the design of the Go creators closely:
- communicate by sharing memory (not share memory by communicating)
- communicate to synchronize (not synchronize to communicate)
Additional note: We use buffered channels to minimize the blockings that occur (since these blockings reduce our efficiency).
In doing all of the above, this concurrent/multi-threaded program highlights emphases on:
- Synchronization
- Threads and Goroutines
- Communication and Channels
The Dining Philosophers always seem to make their way into every concurrency discussion. We are not immune to their visit. It is a classic to illustrate concurrency problems.
Five philosophers -- say, Plato, Aristotle, Socrates, Sun Tzu, and Kant -- walk into a round, dining table and take a seat. They are served each a bowl of rice dish, and they all want to start eating immediately. But only five chopsticks are available (not five pairs).
The five chopsticks are placed such that there is one chopstick between every adjacent pair of philosophers. So they can't all eat at the same time. In code, an object-oriented coder may do the following
- create a philosopher object and a chopstick object.
- add a method Eat() to the philosopher object: each philosopher take the chopstick to his left and then to his right.
- to ensure that no two or more threads access a chopstick at one time, it adds a block to the aforementioned method.
- So the philosopher objects run concurrently, picking the chopstick to his left. And then each one wants to pick the chopstick to his right.
- But interleavings happen where every philosopher holds the chopstick to his left, so when each goes to the next line of instruction to pick up the chopstick to his right, it's blocked.
- Now each philosopher is waiting for the next philosopher to unblock, but to unblock it needs the next one to unblock. There are circular dependencies, and thus a deadlock.
In the end, person1 is waiting for person2 to finish, who can't finish because he has only 1 chopstick. That person 2 is waiting for person 3 to release the left chopstick, which he isn't going to do because he needs it. Person 3 waits for person4, and so on... and person5 waits for person1. It is stuck, and the go runtime throws an error.
So our concurrent program needs to solve this deadlock: the philosophers must all be able to finish the meal in front of them.
Our concurrent program, file #11, does just that: solve the Dining Philosophers problem.
*In the program as is, each philosopher eats only once. However, if you want to see that there really is no deadlock even if the cycle is infinite, you can uncomment the for loop inside the file:
// for {
...
// }
Then the program will run infinitely in your Command Line, until you press ctrl-c.
Main Topics:
- Deadlocks
- Communication
- Synchronization
- Mutex
- Waitgroup