Concurrency ialah asas reka bentuk Go, dan ini adalah salah satu sebab mengapa bahasa itu mendapat begitu populariti. Walaupun kebanyakan pembangun sudah biasa dengan goroutin dan saluran asas, terdapat seluruh dunia corak lanjutan yang menunggu untuk diterokai.
Mari kita mulakan dengan penyegerakan.Cond, primitif penyegerakan berkuasa yang sering diabaikan. Ia amat berguna apabila anda perlu menyelaraskan berbilang goroutin berdasarkan syarat. Berikut ialah contoh mudah:
var count int var mutex sync.Mutex var cond = sync.NewCond(&mutex) func main() { for i := 0; i < 10; i++ { go increment() } time.Sleep(time.Second) cond.Broadcast() time.Sleep(time.Second) fmt.Println("Final count:", count) } func increment() { mutex.Lock() defer mutex.Unlock() cond.Wait() count++ }
Dalam contoh ini, kami menggunakan sync.Cond untuk menyelaraskan berbilang goroutin. Mereka semua menunggu isyarat sebelum menambah kiraan. Corak ini berguna apabila anda perlu menyegerakkan berbilang goroutin berdasarkan keadaan tertentu.
Operasi atom ialah satu lagi alat berkuasa dalam kit alat serentak Go. Mereka membenarkan penyegerakan tanpa kunci, yang boleh meningkatkan prestasi dengan ketara dalam senario tertentu. Begini cara anda boleh menggunakan operasi atom untuk melaksanakan pembilang mudah:
var counter int64 func main() { for i := 0; i < 1000; i++ { go func() { atomic.AddInt64(&counter, 1) }() } time.Sleep(time.Second) fmt.Println("Counter:", atomic.LoadInt64(&counter)) }
Kod ini jauh lebih mudah dan berkemungkinan lebih cekap daripada menggunakan mutex untuk operasi asas sedemikian.
Sekarang, mari kita bincangkan tentang beberapa corak yang lebih kompleks. Corak kipas keluar/masuk kipas ialah cara yang berkesan untuk menyelaraskan kerja. Berikut ialah pelaksanaan mudah:
func fanOut(input <-chan int, workers int) []<-chan int { channels := make([]<-chan int, workers) for i := 0; i < workers; i++ { channels[i] = work(input) } return channels } func fanIn(channels ...<-chan int) <-chan int { var wg sync.WaitGroup out := make(chan int) output := func(c <-chan int) { for n := range c { out <- n } wg.Done() } wg.Add(len(channels)) for _, c := range channels { go output(c) } go func() { wg.Wait() close(out) }() return out } func work(in <-chan int) <-chan int { out := make(chan int) go func() { for n := range in { out <- n * n } close(out) }() return out }
Corak ini membolehkan anda mengagihkan kerja merentasi berbilang goroutine dan kemudian mengumpul hasilnya. Ia amat berguna untuk tugasan terikat CPU yang boleh diselaraskan.
Kumpulan pekerja ialah satu lagi corak biasa dalam pengaturcaraan serentak. Mereka membenarkan anda mengehadkan bilangan goroutine berjalan serentak, yang boleh menjadi penting untuk mengurus penggunaan sumber. Berikut ialah pelaksanaan mudah:
func workerPool(jobs <-chan int, results chan<- int, workers int) { var wg sync.WaitGroup for i := 0; i < workers; i++ { wg.Add(1) go func() { defer wg.Done() for job := range jobs { results <- job * 2 } }() } wg.Wait() close(results) }
Kumpulan pekerja ini memproses kerja serentak, tetapi mengehadkan bilangan operasi serentak kepada bilangan pekerja.
Saluran paip ialah satu lagi corak berkuasa dalam Go. Mereka membenarkan anda memecahkan operasi yang kompleks ke dalam peringkat yang boleh diproses secara serentak. Berikut ialah contoh mudah:
func gen(nums ...int) <-chan int { out := make(chan int) go func() { for _, n := range nums { out <- n } close(out) }() return out } func sq(in <-chan int) <-chan int { out := make(chan int) go func() { for n := range in { out <- n * n } close(out) }() return out } func main() { for n := range sq(sq(gen(2, 3))) { fmt.Println(n) } }
Saluran paip ini menjana nombor, menduakannya, dan kemudian menduakan hasilnya semula. Setiap peringkat berjalan dalam goroutine sendiri, membolehkan pemprosesan serentak.
Penutupan yang menarik adalah penting dalam sistem pengeluaran. Berikut ialah corak untuk melaksanakan penutupan yang anggun:
func main() { done := make(chan struct{}) go worker(done) // Simulate work time.Sleep(time.Second) // Signal shutdown close(done) fmt.Println("Shutting down...") time.Sleep(time.Second) // Give worker time to clean up } func worker(done <-chan struct{}) { for { select { case <-done: fmt.Println("Worker: Cleaning up...") return default: fmt.Println("Worker: Working...") time.Sleep(100 * time.Millisecond) } } }
Corak ini membolehkan pekerja membersihkan dan keluar dengan anggun apabila diberi isyarat.
Pengendalian tamat masa adalah satu lagi aspek penting dalam pengaturcaraan serentak. Kenyataan pilih Go memudahkan perkara ini:
func doWork() <-chan int { ch := make(chan int) go func() { time.Sleep(2 * time.Second) ch <- 42 }() return ch } func main() { select { case result := <-doWork(): fmt.Println("Result:", result) case <-time.After(1 * time.Second): fmt.Println("Timeout!") } }
Kod ini akan tamat masa jika doWork mengambil masa lebih lama daripada satu saat untuk menghasilkan hasil.
Penyebaran pembatalan ialah corak di mana isyarat pembatalan diturunkan melalui rangkaian panggilan fungsi. Pakej konteks dalam Go direka untuk ini:
var count int var mutex sync.Mutex var cond = sync.NewCond(&mutex) func main() { for i := 0; i < 10; i++ { go increment() } time.Sleep(time.Second) cond.Broadcast() time.Sleep(time.Second) fmt.Println("Final count:", count) } func increment() { mutex.Lock() defer mutex.Unlock() cond.Wait() count++ }
Corak ini membolehkan pembatalan mudah operasi yang berjalan lama.
Sekarang, mari lihat beberapa contoh dunia sebenar. Berikut ialah pelaksanaan mudah pengimbang beban:
var counter int64 func main() { for i := 0; i < 1000; i++ { go func() { atomic.AddInt64(&counter, 1) }() } time.Sleep(time.Second) fmt.Println("Counter:", atomic.LoadInt64(&counter)) }
Pengimbang beban ini mengedarkan permintaan kepada pelayan yang paling sedikit dimuatkan, mengemas kini beban dalam masa nyata.
Penghadan kadar ialah satu lagi keperluan biasa dalam sistem teragih. Berikut ialah pelaksanaan baldi token yang mudah:
func fanOut(input <-chan int, workers int) []<-chan int { channels := make([]<-chan int, workers) for i := 0; i < workers; i++ { channels[i] = work(input) } return channels } func fanIn(channels ...<-chan int) <-chan int { var wg sync.WaitGroup out := make(chan int) output := func(c <-chan int) { for n := range c { out <- n } wg.Done() } wg.Add(len(channels)) for _, c := range channels { go output(c) } go func() { wg.Wait() close(out) }() return out } func work(in <-chan int) <-chan int { out := make(chan int) go func() { for n := range in { out <- n * n } close(out) }() return out }
Penghad kadar ini membenarkan bilangan operasi tertentu sesaat, melancarkan ledakan trafik.
Baris gilir tugas yang diedarkan ialah kes penggunaan biasa untuk ciri serentak Go. Berikut ialah pelaksanaan mudah:
func workerPool(jobs <-chan int, results chan<- int, workers int) { var wg sync.WaitGroup for i := 0; i < workers; i++ { wg.Add(1) go func() { defer wg.Done() for job := range jobs { results <- job * 2 } }() } wg.Wait() close(results) }
Baris gilir tugas yang diedarkan ini membolehkan berbilang pekerja memproses tugasan secara serentak.
Waktu jalan Go menyediakan alat yang berkuasa untuk mengurus goroutin. Fungsi GOMAXPROCS membolehkan anda mengawal bilangan urutan OS yang boleh melaksanakan kod Go secara serentak:
func gen(nums ...int) <-chan int { out := make(chan int) go func() { for _, n := range nums { out <- n } close(out) }() return out } func sq(in <-chan int) <-chan int { out := make(chan int) go func() { for n := range in { out <- n * n } close(out) }() return out } func main() { for n := range sq(sq(gen(2, 3))) { fmt.Println(n) } }
Ini menetapkan bilangan urutan OS kepada bilangan CPU, yang boleh meningkatkan prestasi untuk tugas terikat CPU.
Mengoptimumkan kod serentak selalunya melibatkan pengimbangan antara keselarian dan overhed mencipta dan mengurus goroutin. Alat pemprofilan seperti pprof boleh membantu mengenal pasti kesesakan:
func main() { done := make(chan struct{}) go worker(done) // Simulate work time.Sleep(time.Second) // Signal shutdown close(done) fmt.Println("Shutting down...") time.Sleep(time.Second) // Give worker time to clean up } func worker(done <-chan struct{}) { for { select { case <-done: fmt.Println("Worker: Cleaning up...") return default: fmt.Println("Worker: Working...") time.Sleep(100 * time.Millisecond) } } }
Kod ini membolehkan pprof, membolehkan anda membuat profil kod serentak anda dan mengenal pasti isu prestasi.
Kesimpulannya, ciri konkurensi Go menyediakan kit alat yang berkuasa untuk membina sistem yang cekap dan berskala. Dengan menguasai corak dan teknik lanjutan ini, anda boleh memanfaatkan sepenuhnya pemproses berbilang teras moden dan membina aplikasi yang mantap dan berprestasi tinggi. Ingat, concurrency bukan hanya tentang kelajuan - ia tentang mereka bentuk kod yang bersih dan boleh diurus yang boleh mengendalikan senario dunia sebenar yang kompleks. Jadi majulah dan tempuhi cabaran serentak itu!
Pastikan anda melihat ciptaan kami:
Pusat Pelabur | Hidup Pintar | Epos & Gema | Misteri Membingungkan | Hindutva | Pembangunan Elit | Sekolah JS
Tech Koala Insights | Dunia Epok & Gema | Medium Pusat Pelabur | Medium Misteri Membingungkan | Sains & Zaman Sederhana | Hindutva Moden
Atas ialah kandungan terperinci Menguasai Go's Advanced Concurrency: Tingkatkan Kuasa dan Prestasi Kod Anda. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!