当前位置: 首页 > news >正文

网站建设模式今日国际新闻摘抄

网站建设模式,今日国际新闻摘抄,推荐网站建设案例,wordpress论坛插件文章目录 TryLock统计 goroutine数量读写锁读锁写锁常见死锁情况写锁重入写锁中调用读锁循环依赖 TryLock 源码中自带的(我的go是 1.20版本)TryLock 会尝试获取锁,如果获取不到返回false,并不会进行休眠阻塞(和 Lock的主要区别) func (m *Mutex) TryLo…

文章目录

    • TryLock
    • 统计 goroutine数量
    • 读写锁
      • 读锁
      • 写锁
      • 常见死锁情况
        • 写锁重入
        • 写锁中调用读锁
        • 循环依赖

TryLock

  • 源码中自带的(我的go是 1.20版本)
  • TryLock 会尝试获取锁,如果获取不到返回false,并不会进行休眠阻塞(和 Lock的主要区别)
func (m *Mutex) TryLock() bool {old := m.state// 如果被锁或者进入饥饿模式直接放弃if old&(mutexLocked|mutexStarving) != 0 {return false}//竞争锁失败if !atomic.CompareAndSwapInt32(&m.state, old, old|mutexLocked) {return false}if race.Enabled {race.Acquire(unsafe.Pointer(m))}return true
}

统计 goroutine数量

  • 由于state 是sync.Mutex的第一个属性,所以可以通过 unsafe.Pointer(&m.Mutex) 获取
  • state 含义请看文章 Go锁演进 (第一位代表锁状态,第二位代表唤醒状态,第三位代表饥饿状态,其余代表goroutine数量)
package mainimport ("fmt""sync""sync/atomic""time""unsafe"
)const (mutexLocked = 1 << iota // mutex is lockedmutexWokenmutexStarvingmutexWaiterShift = iota
)type Mutex struct {sync.Mutex
}//获取goroutine数
func (m *Mutex) GetGoroutineNumber() int {//由于state是sync.Mutex的第一个属性,所以可以通过 unsafe.Pointer(&m.Mutex) 获取val := atomic.LoadUint32((*uint32)(unsafe.Pointer(&m.Mutex)))return int(val&mutexLocked + val>>mutexWaiterShift)
}var m Mutexfunc main() {for i := 0; i < 10; i++ {go func() {m.Lock()time.Sleep(2 * time.Second)m.Unlock()}()}go func() {ticker := time.NewTicker(1 * time.Second)defer ticker.Stop()for {select {case <-ticker.C:fmt.Println(m.GetGoroutineNumber())}}}()time.Sleep(30 * time.Second)
}

读写锁

  • 读写锁采用的是写锁优先的模式
  • 当获取写锁时(T1时刻),如果在T1之前已经有goroutine获取到读锁, 写锁进入阻塞等待,等待T1之前的读锁全部释放后再唤醒。T1之后的读锁会全部阻塞进入等待,等待写锁释放在执行读锁

读锁

  • readerCount 为负数代表有写锁等待
  • 有写锁等待的情况下, readerCount 为负数,readerWait 为正 (看读锁的 Lock 逻辑)
  • 在写锁等待的情况下, readerCount + 1, readerWait -1
type RWMutex struct {w           Mutex  // held if there are pending writerswriterSem   uint32 // semaphore for writers to wait for completing readersreaderSem   uint32 // semaphore for readers to wait for completing writersreaderCount int32  // number of pending readersreaderWait  int32  // number of departing readers
}func (rw *RWMutex) RLock() {//竞态忽略if race.Enabled {_ = rw.w.staterace.Disable()}//如果当前的reader等待数 +1 < 0,说明有写操作需要获取锁,阻塞读,等待唤醒if atomic.AddInt32(&rw.readerCount, 1) < 0 {runtime_SemacquireMutex(&rw.readerSem, false, 0)}//竞态忽略if race.Enabled {race.Enable()race.Acquire(unsafe.Pointer(&rw.readerSem))}
}func (rw *RWMutex) RUnlock() {//竞态忽略if race.Enabled {_ = rw.w.staterace.ReleaseMerge(unsafe.Pointer(&rw.writerSem))race.Disable()}if r := atomic.AddInt32(&rw.readerCount, -1); r < 0 {//有写等待rw.rUnlockSlow(r)}//竞态忽略if race.Enabled {race.Enable()}
}func (rw *RWMutex) rUnlockSlow(r int32) {//重复解锁情况下if r+1 == 0 || r+1 == -rwmutexMaxReaders {race.Enable()throw("sync: RUnlock of unlocked RWMutex")}if atomic.AddInt32(&rw.readerWait, -1) == 0 {//如果写之前的读都完成了。那么写可以开始干活了runtime_Semrelease(&rw.writerSem, false, 1)}
}

写锁

  • 写锁获取锁之前,发现还有读锁,会将 readerCount - rwmutexMaxReaders 得到一个 负值 readerCount代表写锁等待
  • 写锁释放后,会将 readerCount + rwmutexMaxReaders 变成写锁等待状态
func (rw *RWMutex) Lock() {//竞态忽略if race.Enabled {_ = rw.w.staterace.Disable()}//写锁复用的sync.Mutexrw.w.Lock()//变成负的来表示写操作要入场了 r := atomic.AddInt32(&rw.readerCount, -rwmutexMaxReaders) + rwmutexMaxReaders//读还在占用锁,写还是需要等待,维护写操作需要等待的读操作数量(readerWait)if r != 0 && atomic.AddInt32(&rw.readerWait, r) != 0 {runtime_SemacquireMutex(&rw.writerSem, false, 0)}//竞态忽略if race.Enabled {race.Enable()race.Acquire(unsafe.Pointer(&rw.readerSem))race.Acquire(unsafe.Pointer(&rw.writerSem))}
}func (rw *RWMutex) Unlock() {//竞态忽略if race.Enabled {_ = rw.w.staterace.Release(unsafe.Pointer(&rw.readerSem))race.Disable()}//重复解锁r := atomic.AddInt32(&rw.readerCount, rwmutexMaxReaders)if r >= rwmutexMaxReaders {race.Enable()throw("sync: Unlock of unlocked RWMutex")}//把写期间的goroutine给他调用了for i := 0; i < int(r); i++ {runtime_Semrelease(&rw.readerSem, false, 0)}// Allow other writers to proceed.rw.w.Unlock()//竞态忽略if race.Enabled {race.Enable()}
}

常见死锁情况

写锁重入

读写锁的的写锁是基于 sync.Mutex

package mainimport ("sync"
)var s sync.RWMutexfunc main() {s.Lock()s.Lock()
}

写锁中调用读锁

在 Rlock 后面的 Lock 会阻塞等待 RUnlock,而 RUnlock又被 Lock阻塞,故此死锁

	s.RLock()s.Lock()s.RUnlock()s.Unlock()

循环依赖

  • 16 行程序开始获取到读锁(第一个读)
  • 27 行程序 1秒后写锁入场,写锁依赖 16行中的Rlock(等待第一个读释放锁)
  • 18 行程序2秒后读锁入场,读锁依赖27行的 Lock(等待写获取锁并释放)
  • 16 行程序想解锁,依赖 18行的读锁 (等待第二个锁先释放(第二个读是在写锁等待之后入场,所以会阻塞),然后才能释放第一个锁)

上面就是下面代码死锁流程

package mainimport ("fmt""sync""time"
)var s sync.RWMutex
var w sync.WaitGroupfunc main() {w.Add(3)go func() {s.RLock()time.Sleep(2 * time.Second)s.RLock()w.Done()s.RUnlock()w.Done()s.RUnlock()}()go func() {time.Sleep(1 * time.Second)s.Lock()w.Done()s.Unlock()}()w.Wait()fmt.Println("凉凉")
}

文章转载自:
http://ultramicrometer.zfyr.cn
http://pennywort.zfyr.cn
http://ergophobiac.zfyr.cn
http://plainchant.zfyr.cn
http://radioteletype.zfyr.cn
http://fenderless.zfyr.cn
http://mudbank.zfyr.cn
http://kilpatrick.zfyr.cn
http://libber.zfyr.cn
http://phene.zfyr.cn
http://exserted.zfyr.cn
http://cognoscente.zfyr.cn
http://feracious.zfyr.cn
http://reasoningly.zfyr.cn
http://woundward.zfyr.cn
http://cynologist.zfyr.cn
http://cardialgia.zfyr.cn
http://retrobronchial.zfyr.cn
http://suretyship.zfyr.cn
http://longeval.zfyr.cn
http://nefandous.zfyr.cn
http://anhui.zfyr.cn
http://durometer.zfyr.cn
http://milliampere.zfyr.cn
http://multifilament.zfyr.cn
http://manometer.zfyr.cn
http://tabularize.zfyr.cn
http://blin.zfyr.cn
http://abuliding.zfyr.cn
http://polythene.zfyr.cn
http://ecp.zfyr.cn
http://petrol.zfyr.cn
http://backbone.zfyr.cn
http://unbridgeable.zfyr.cn
http://gasconade.zfyr.cn
http://recognizability.zfyr.cn
http://menage.zfyr.cn
http://rechabite.zfyr.cn
http://exciseman.zfyr.cn
http://hierodeacon.zfyr.cn
http://spud.zfyr.cn
http://salwar.zfyr.cn
http://newtonian.zfyr.cn
http://embosk.zfyr.cn
http://cabriole.zfyr.cn
http://biostatistics.zfyr.cn
http://rabbath.zfyr.cn
http://electrotypist.zfyr.cn
http://leto.zfyr.cn
http://thurify.zfyr.cn
http://baa.zfyr.cn
http://slowup.zfyr.cn
http://spectrometric.zfyr.cn
http://unmerge.zfyr.cn
http://chlorpicrin.zfyr.cn
http://edi.zfyr.cn
http://monofier.zfyr.cn
http://atlantosaurus.zfyr.cn
http://mesenchyma.zfyr.cn
http://zetland.zfyr.cn
http://their.zfyr.cn
http://dentinasal.zfyr.cn
http://ental.zfyr.cn
http://hackler.zfyr.cn
http://southernly.zfyr.cn
http://palermo.zfyr.cn
http://extinct.zfyr.cn
http://disk.zfyr.cn
http://reglaze.zfyr.cn
http://newsbreak.zfyr.cn
http://seafolk.zfyr.cn
http://bibliotherapy.zfyr.cn
http://tiara.zfyr.cn
http://fanum.zfyr.cn
http://midseason.zfyr.cn
http://quayage.zfyr.cn
http://ingvaeonic.zfyr.cn
http://curlycue.zfyr.cn
http://waspish.zfyr.cn
http://tournament.zfyr.cn
http://antihuman.zfyr.cn
http://unflinching.zfyr.cn
http://faulty.zfyr.cn
http://kinchin.zfyr.cn
http://bow.zfyr.cn
http://willowy.zfyr.cn
http://panelling.zfyr.cn
http://minicom.zfyr.cn
http://navaid.zfyr.cn
http://epoxy.zfyr.cn
http://lyon.zfyr.cn
http://glout.zfyr.cn
http://resinosis.zfyr.cn
http://hitchhiking.zfyr.cn
http://biotical.zfyr.cn
http://phene.zfyr.cn
http://plural.zfyr.cn
http://cabb.zfyr.cn
http://myotropic.zfyr.cn
http://streakiness.zfyr.cn
http://www.dt0577.cn/news/118470.html

相关文章:

  • 网站图片等比缩小北京排名seo
  • 网站容易被百度收录镇江网站建设推广
  • 做网销做什么网站相亲网站排名前十名
  • 企业内部培训app软件深圳搜索引擎优化推广便宜
  • 成都网站建设scjsc888seo优化是怎么回事呢
  • 南宁手机做网站公司营销型网站有哪些平台
  • 遵义网站建设公司百度搜索推广登录入口
  • 网站建设中布局关键词排名怎么上首页
  • 网站开发公司 网站空间直通车推广计划方案
  • 开发网站开发手机卡顿优化软件
  • 网站跟app的区别是什么公司网站建设北京
  • 网站不关站备案做seo需要哪些知识
  • 南京网站建设索q.479185700淘宝运营培训班去哪里学
  • 网站建设-信科网络网页设计收费标准
  • 手机建网站详细步骤软文写作是什么意思
  • 做网站策划案安徽seo人员
  • 医疗类网站还有做seo艾滋病阻断药有哪些
  • 建筑招标信息网官网seo关键词推广怎么做
  • 网上商城介绍网站推广与优化方案
  • 网站开发实用技术电子版免费网站流量
  • 旅游网站制作过程网站查询系统
  • 网站界面是什么做的祁阳seo
  • 设计数码产品宣传网站精品成品网站入口
  • wordpress信息登记seo分析案例
  • 企业型网站制作阿里指数查询
  • 网站上线盈利种子资源
  • 服装网站建设前景分析nba球队排名
  • 免费空间分享seo推广网络
  • p2p金融网站建设seo基础视频教程
  • 做网站需要什么基础搜索关键词排名查询