go系列 鎖的初識

Go基礎之鎖的初識

當我們的程序就一個線程的時候是不需要用到鎖的,但是通常我們實際的代碼不會是單個線程的,所有這個時候就需要用到鎖了,那么關于鎖的使用場景主要涉及到哪些呢?

  • 當我們多個線程在讀相同的數據的時候則是需要加鎖的
  • 當我們的程序既有讀又有寫的時候更是需要加鎖的
  • 當我們有多個線程在寫的時候同樣也是需要加鎖

互斥鎖

互斥鎖:同一個時刻只有一個線程能夠拿到鎖

我們先通過一個例子來演示,如果當多個線程同時更改一個變量,結果會是怎么樣
不加鎖版本

復制代碼
package mainimport ("sync""fmt"
)var (//lock sync.Mutexcount intw sync.WaitGroup  //用于等待子線程執行完之后退出
)func main() {w.Add(1) // 在調用線程前執行w.addgo func(){for i:=0;i<100000;i++{count++}w.Done()  //執行完 執行w.Done}()for i :=0;i<100000;i++{count++}w.Wait() // 最后執行w.wait等待所有的線程執行完畢fmt.Println(count)}
復制代碼

當我們運行多次就可以發現,最后的結果基本不可能是我們先看到的:200000
我們修改代碼代碼需要加鎖保護的地方加上鎖,并且這里加的是互斥鎖,修改后的代碼為:

復制代碼
package mainimport ("sync""fmt"
)var (lock sync.Mutexcount intw sync.WaitGroup  //用于等待子線程執行完之后退出
)func main() {w.Add(1) // 在調用線程前執行w.addgo func(){for i:=0;i<100000;i++{lock.Lock()count++lock.Unlock()}w.Done()  //執行完 執行w.Done}()for i :=0;i<100000;i++{lock.Lock()count++lock.Unlock()}w.Wait() // 最后執行w.wait等待所有的線程執行完畢fmt.Println(count)}
復制代碼

這次當我們多次運行的時候,就能保證我們每次都能看到我們想要的值:200000
接下來看讀寫鎖

讀寫鎖

讀寫鎖主要用到讀多寫少的場景
讀寫鎖分為:讀鎖和寫鎖

如果自己設置了一個寫鎖,那么其他讀的線程以及寫的線程都拿不到鎖,這個時候和互斥鎖的功能相同
如果自己設置了一個讀鎖,那么其他寫的線程是拿不到鎖的,但是其他讀的線程都是可以拿到這個鎖

我們把上面的例子代碼進行更改:

復制代碼
package mainimport ("sync""fmt"
)    var (rwlock sync.RWMutexw sync.WaitGroupcount int
)func main() {w.Add(1)go func(){for i:=0;i<1000000;i++{rwlock.Lock() // 這里定義了一個寫鎖count++rwlock.Unlock()}w.Done()}()for i:=0;i<1000000;i++{rwlock.Lock() // 這里定義了一個寫鎖count++rwlock.Unlock()}w.Wait()fmt.Println(count)
}
復制代碼

通過設置寫鎖,我們同樣可以實現數據的一致性
下面是一個讀鎖的使用例子:

復制代碼
package mainimport ("sync""fmt"
)var (rwlock sync.RWMutexw sync.WaitGroupcount int
)func main() {w.Add(1)go func(){for i:=0;i<1000000;i++{rwlock.Lock() // 這里定義了一個寫鎖count++rwlock.Unlock()}w.Done()}()for i:=0;i<16;i++{w.Add(1)go func(){rwlock.RLock() //這里定義了一個讀鎖fmt.Println(count)rwlock.RUnlock() //釋放讀鎖w.Done()}()}w.Wait()fmt.Println(count)
}
復制代碼

Go中的原子操作

原子操作,我們則不需加鎖,也能保證數據的一致性
并且如果只是計算,那么原子操作則是最快的

實例代碼:

復制代碼
package mainimport ("sync"//"time""sync/atomic""fmt"
)var (w sync.WaitGroupcount int32
)func main() {w.Add(1)//start := time.Now().UnixNano()go func() {for i:=0;i<1000000;i++{atomic.AddInt32(&count,1)}w.Done()}()for i:=0;i<1000000;i++{atomic.AddInt32(&count,1)}w.Wait()//end := time.Now().UnixNano()//fmt.Println((end- start)/1000/1000)fmt.Println(count)
}
復制代碼

?

所有的努力都值得期許,每一份夢想都應該灌溉!

轉載于:https://www.cnblogs.com/flying1819/articles/8832749.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/250521.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/250521.shtml
英文地址,請注明出處:http://en.pswp.cn/news/250521.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Java 試題六

Java 試題六 1、Collection 和 Collections的區別 答&#xff1a;Collection是集合類的上級接口&#xff0c;繼承與他的接口主要有Set 和List。Collections是針對集合類的一個幫助類&#xff0c;他提供一系列靜態方法實現對各種集合的搜索、排序、線程安全化等操作。 2、Set里…

node --- 實現session認證.

跨域認證的問題 互聯網服務離不開用戶認證.一般流程如下: 1、用戶向服務器發送用戶名和密碼。 2、服務器驗證通過后&#xff0c;在當前對話&#xff08;session&#xff09;里面保存相關數據&#xff0c;比如用戶角色、登錄時間等等。 3、服務器向用戶返回一個 session_id&…

回信,我的好朋友王一涵

好了&#xff0c;不拖了&#xff0c;沏一杯咖啡&#xff0c;把信寫完。因為再拿好吃的賄賂你&#xff0c;賄賂不起了—— 一個胖子可以吃窮我的。 王一涵凹&#xff0c;不得了不得了。微胖肉質女生&#xff0c;關于體重我就不提了&#xff0c;只有我知道嘿嘿嘿&#xff0c;在我…

編寫基于Property-based的單元測試

編寫基于Property-based的單元測試 作為一個開發者&#xff0c;你可能認為你的職責就是編寫代碼從而完成需求。我不敢茍同&#xff0c;開發者的工作是通過軟件來解決現實需求&#xff0c;編寫代碼只是軟件開發的其中一個方面&#xff0c;編寫可靠的軟件和產出有價值的代碼更加重…

樹鏈剖分+線段樹 單點修改 區間求和 模板

馬上要去西安打邀請賽了&#xff0c;存下板子 首先是vector存圖的&#xff1a; #include<bits/stdc.h> using namespace std; #define ll long long #define lson l,m,rt<<1 #define rson m1,r,rt<<1|1 #define mid int m (l r) >> 1 const int M …

koa --- seesion實現登錄鑒權

koa vue session 實現一個簡單的登錄邏輯 /login component/login-session.html <!DOCTYPE html><head><script src"https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script><script src"https://unpkg.com/axios/dist/axios.…

BZOJ2216: [Poi2011]Lightning Conductor

第一道此類的題&#xff0c;所以這是一篇假的博客&#xff0c;定理不會證明不理性 也不一定對 我是從這篇博客看的 很顯然是讓你求 p[i] max{a[j] sqrt(i - j)} - a[i] 就是 max{a[j] sqrt(|i - j|)} 這是一個 1D/1D 動態規劃 考慮對于絕對值的情況不好做&#xff0c;那就…

HNOI2018游記

HNOI2018游記 day 0 上午稍微寫了下題保持手感,然后看了一下套路,感覺不會的還是不會. 下午去劃水在湖面上被吹成傻逼... 感覺沒有聯賽前那么緊張了,應該是聯賽考掛了的原因吧.. day1 早上大概7:40就到了考場,和同學聊了一會兒天,看了看配置就進去了. 進去之后敲配置沒有一遍對…

Java 試題七

Java 試題七 1、java中有幾種類型的流&#xff1f;JDK為每種類型的流提供了一些抽象類以供繼承&#xff0c;請說出他們分別是哪些類&#xff1f; 答&#xff1a;字節流&#xff0c;字符流。 字節流繼承于InputStream、OutputStream&#xff0c; 字符流繼承于Reader、Writer…

flume快速入門及應用

? Flume 簡介? Flume 的安裝與配置? Fumne 部署   Flume 是 Cloudera 提供的一個高可用、 高可靠、 分布式的海量日志采集、 聚合和傳輸的系統。 Flume 支持定制各類數據源如 Avro、 Thrift、 Spooling 等。 同時 Flume提供對數據的簡單處理&#xff0c; 并將數據處理結果…

koa --- jwt實現最簡單的Token認證

HTML 有如下html: 先看代碼后挑重點來說明: <!DOCTYPE html><head><script src"https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script><script src"https://unpkg.com/axios/dist/axios.min.js"></script></…

python基礎之常用的高階函數

前言 高階函數指的是能接收函數作為參數的函數或類&#xff1b;python中有一些內置的高階函數&#xff0c;在某些場合使用可以提高代碼的效率&#xff0e; map() map函數可以把一個迭代對象轉換成另一個可迭代對象&#xff0c;不過在python3中&#xff0c;結果都是一個map對象&…

Java 試題八

Java 試題八 1、java中有幾種方法可以實現一個線程&#xff1f;用什么關鍵字修飾同步方法? stop()和suspend()方法為何不推薦使用&#xff1f; 答&#xff1a;有兩種實現方法&#xff0c;分別是繼承Thread類與實現Runnable接口&#xff1b;用synchronized關鍵字修飾同步方法…

bzoj2957 奧妙重重的線段樹

https://www.lydsy.com/JudgeOnline/problem.php?id2957 線段樹的query和update竟然還可以結合起來用&#xff01; 題意&#xff1a;小A的樓房外有一大片施工工地&#xff0c;工地上有N棟待建的樓房。每天&#xff0c;這片工地上的房子拆了又建、建了又拆。他經常無聊地看著窗…

koa --- 使用Github OAuth登錄

準備 登錄github選擇右上角的setting Developer settings -> OAuth Apps -> Register a new application 填入基本信息 點擊綠色的按鈕,可以看見 client_id 和 client secret 理清思路: 開始時,一個登錄的連接,點擊連接.后臺監聽登錄(/login)路由,然后重定向到github…

[數據結構] - ArrayList探究

一 概述 ArrayList可以理解為動態數組&#xff0c;與java的數組相比&#xff0c;它的容量能動態曾長&#xff0c;ArrayList是List接口的可變數組的實現&#xff0c;允許包括null值在內的所有元素。除了實現List接口外&#xff0c;此類還提供一些方法來操作內部用來存儲列表的數…

10.10考試題

voteplus 【問題描述】 R 君博客上有?個投票板塊&#xff0c;?家可以使?投票的?式來表達??對某些問題的贊成或反對的意見。 投票結果是公開的&#xff0c;但是 R 君會把這個結果化成?個最簡分數&#xff0c;如 1:2,4:3。 注意到同?個最簡分數可能代表了不同的總?數&am…

koa --- 跨域,解析POST參數、路由配置

目標 將開發中經常遇見的問題寫在這里方便查詢. 使用Koa創建一個簡單的服務器 const Koa require("koa"); const app new Koa(); app.listen(3000, () >{console.log("[server] Server is running at http://localhost:3000") })使用koa2-cors解決…

mysql數據庫常用操作

目前最流行的數據庫&#xff1a; oracle、mysql、sqlserver、db2、sqline --&#xff1a;單行注釋 #&#xff1a;也是單行注釋 /* 注釋內容*/&#xff1a;多行注釋 mysql -uroot -p密碼&#xff1a;登錄mysql service mysqld restart重啟mysql /etc/my.cnfmysql的配置文件 /var…

數碼相機控制點的自動定位檢校

為簡化控制場相機檢校中的人工量測控制點的繁瑣工作,提高相機檢校精度,本文提出一種方法:只需均勻量測少量控制點的像方坐標獲取相機檢校初始參數,便可通過動態模板匹配實現單影像相機檢校的控制點高精度自動定位檢校。實驗證明此方法檢校精度與人工量測檢校精度相近。 https:/…