What are WaitGroups in go language? how to use?
What are WaitGroups? The following article will take you to understand WaitGroups in the go language and introduce how to use WaitGroups. I hope it will be helpful to you!
#What are WaitGroups?
WaitGroups
is an efficient way to synchronize your goroutines. Imagine you are traveling by car with your family. Your dad stops at a strip mall or fast food restaurant to buy some food and use the bathroom. You'd better want to wait until everyone gets back before driving to Horizon. WaitGroups
helps you do this.
WaitGroups
is defined by calling the sync
package in the standard library.
var wg sync.WaitGroup
So, what is WaitGroup
? WaitGroup
is a structure that contains certain information about how many goroutine
the program needs to wait for. It is a group containing the number of goroutines
you need to wait for.
WaitGroups has three most important methods: Add
, Done
and Wait
.
- Add: Add to the total number of goroutines you need to wait for.
- Done: Subtract one from the total number of goroutines you need to wait for.
- Wait: Blocks the code from continuing until there are no more goroutines to wait for.
How to use WaitGroups
Let’s take a look at a piece of code:
package main import ( "fmt" "sync" "time" ) func main() { var wg sync.WaitGroup wg.Add(1) go func() { defer wg.Done() fmt.Println(time.Now(), "start") time.Sleep(time.Second) fmt.Println(time.Now(), "done") }() wg.Wait() fmt.Println(time.Now(), "exiting...") }
2022-08-21 17:01:54.184744229 +0900 KST m=+0.000021800 start 2022-08-21 17:01:55.184932851 +0900 KST m=+1.000210473 done 2022-08-21 17:01:55.18507731 +0900 KST m=+1.000354912 exiting...
- We first initialize a
WaitGroup Instance of wg
. - Then we add 1 to
wg
because we want to wait for agoroutine
to complete. - Then we run this
goroutine
. Inside thegoroutine
, we make a delayed call towg.Done()
to ensure that we decrement the number ofgoroutine
to wait for. If we don't do this, then the code will wait forever for thegoroutine
to complete and will cause a deadlock. - After the
goroutine
call, we want to make sure to block the code untilWaitGroup
is empty. We do this by callingwg.Wait()
.
Why use WaitGroups instead of channels?
Now that we know how to use WaitGroups, a natural thought leads us to this question: Why use WaitGroups instead of channels?
Based on my experience, there are several reasons.
-
WaitGroups
tends to be more intuitive. When you read a piece of code, when you see aWaitGroup
, you immediately know what the code is doing. The method names are clear and get to the point. However, with channels, sometimes it's not so clear. Using channels is smart, but when you read a complex piece of code, it can be cumbersome to understand. - Sometimes, you don't need to use channels. For example, let's take a look at this code:
var wg sync.WaitGroup for i := 0; i < 5; i++ { wg.Add(1) go func() { defer wg.Done() fmt.Println(time.Now(), "start") time.Sleep(time.Second) fmt.Println(time.Now(), "done") }() } wg.Wait() fmt.Println(time.Now(), "exiting...")
You can see that this goroutine
does not communicate data with other goroutine
. If your goroutine
is a one-time job and you don't need to know the result, using WaitGroup
is preferable. Now look at this code:
ch := make(chan int) for i := 0; i < 5; i++ { go func() { randomInt := rand.Intn(10) ch <- randomInt }() } for i := 0; i < 5; i++ { fmt.Println(<-ch) }
Here, goroutine
is sending data to channel
. In these cases we don't need to use WaitGroup
as that would be redundant. If the receive has already done enough blocking, why wait for the goroutine
to complete?
WaitGroups
is specially used to handle waiting for goroutines
. I think the main purpose of channels is to communicate data. You can't use WaitGroup
to send and receive data, but you can use a channel
to synchronize your goroutines
.
Finally, there is no right answer. I know this can be annoying, but it depends on you and the team you work for. Whatever method is best, no answer is wrong. I personally prefer to use WaitGroups
for synchronization, but your situation may be different. Choose what feels most intuitive to you.
One thing to note
Sometimes, you may need to pass a WaitGroup
instance to a goroutine
. There may be several WaitGroup
to handle different goroutine
, or it may be a design choice. Whatever the reason, make sure to pass a pointer to WaitGroup
, like this:
var wg sync.WaitGroup for i := 0; i < 5; i++ { wg.Add(1) go func(wg *sync.WaitGroup) { defer wg.Done() fmt.Println(time.Now(), "start") time.Sleep(time.Second) fmt.Println(time.Now(), "done") }(&wg) } wg.Wait() fmt.Println(time.Now(), "exiting...")
The reason is that Go is a pass-by-value language. This means that whenever you pass an argument to a function, Go copies the argument and passes it instead of the original object. What happens in this case is that the entire WaitGroup
object will be copied, which means that the goroutine
will handle a completely different WaitGroup. wg.Done()
does not subtract from the original wg, but subtracts a copy of it, which only exists in goroutine
.
Summary
By using WaitGroups
, we can easily synchronize goroutines
, ensuring that our code is executed at the correct time. Although channels can also be used for synchronization, WaitGroups
are generally more intuitive and easier to read. When using WaitGroup
, be sure to pass the pointer to WaitGroup
correctly to prevent copy issues. No matter which method you choose, choose the one that's most intuitive and works best for you and your team.
Recommended learning: Golang tutorial
The above is the detailed content of What are WaitGroups in go language? how to use?. For more information, please follow other related articles on the PHP Chinese website!

Mastering the strings package in Go language can improve text processing capabilities and development efficiency. 1) Use the Contains function to check substrings, 2) Use the Index function to find the substring position, 3) Join function efficiently splice string slices, 4) Replace function to replace substrings. Be careful to avoid common errors, such as not checking for empty strings and large string operation performance issues.

You should care about the strings package in Go because it simplifies string manipulation and makes the code clearer and more efficient. 1) Use strings.Join to efficiently splice strings; 2) Use strings.Fields to divide strings by blank characters; 3) Find substring positions through strings.Index and strings.LastIndex; 4) Use strings.ReplaceAll to replace strings; 5) Use strings.Builder to efficiently splice strings; 6) Always verify input to avoid unexpected results.

ThestringspackageinGoisessentialforefficientstringmanipulation.1)Itofferssimpleyetpowerfulfunctionsfortaskslikecheckingsubstringsandjoiningstrings.2)IthandlesUnicodewell,withfunctionslikestrings.Fieldsforwhitespace-separatedvalues.3)Forperformance,st

WhendecidingbetweenGo'sbytespackageandstringspackage,usebytes.Bufferforbinarydataandstrings.Builderforstringoperations.1)Usebytes.Bufferforworkingwithbyteslices,binarydata,appendingdifferentdatatypes,andwritingtoio.Writer.2)Usestrings.Builderforstrin

Go's strings package provides a variety of string manipulation functions. 1) Use strings.Contains to check substrings. 2) Use strings.Split to split the string into substring slices. 3) Merge strings through strings.Join. 4) Use strings.TrimSpace or strings.Trim to remove blanks or specified characters at the beginning and end of a string. 5) Replace all specified substrings with strings.ReplaceAll. 6) Use strings.HasPrefix or strings.HasSuffix to check the prefix or suffix of the string.

Using the Go language strings package can improve code quality. 1) Use strings.Join() to elegantly connect string arrays to avoid performance overhead. 2) Combine strings.Split() and strings.Contains() to process text and pay attention to case sensitivity issues. 3) Avoid abuse of strings.Replace() and consider using regular expressions for a large number of substitutions. 4) Use strings.Builder to improve the performance of frequently splicing strings.

Go's bytes package provides a variety of practical functions to handle byte slicing. 1.bytes.Contains is used to check whether the byte slice contains a specific sequence. 2.bytes.Split is used to split byte slices into smallerpieces. 3.bytes.Join is used to concatenate multiple byte slices into one. 4.bytes.TrimSpace is used to remove the front and back blanks of byte slices. 5.bytes.Equal is used to compare whether two byte slices are equal. 6.bytes.Index is used to find the starting index of sub-slices in largerslices.

Theencoding/binarypackageinGoisessentialbecauseitprovidesastandardizedwaytoreadandwritebinarydata,ensuringcross-platformcompatibilityandhandlingdifferentendianness.ItoffersfunctionslikeRead,Write,ReadUvarint,andWriteUvarintforprecisecontroloverbinary


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

SublimeText3 Chinese version
Chinese version, very easy to use

VSCode Windows 64-bit Download
A free and powerful IDE editor launched by Microsoft

SecLists
SecLists is the ultimate security tester's companion. It is a collection of various types of lists that are frequently used during security assessments, all in one place. SecLists helps make security testing more efficient and productive by conveniently providing all the lists a security tester might need. List types include usernames, passwords, URLs, fuzzing payloads, sensitive data patterns, web shells, and more. The tester can simply pull this repository onto a new test machine and he will have access to every type of list he needs.

Notepad++7.3.1
Easy-to-use and free code editor

SAP NetWeaver Server Adapter for Eclipse
Integrate Eclipse with SAP NetWeaver application server.
