Go语言并发编程-同步和锁

同步和锁

概述

同步是并发编程的基本要素之一,我们通过channel可以完成多个goroutine间数据和信号的同步。

除了channel外,我们还可以使用go的官方同步包sync,sync/atomic 完成一些基础的同步功能。主要包含同步数据、锁、原子操作等。

一个同步失败的示例:

 func SyncErr() {wg := sync.WaitGroup{}// 计数器counter := 0// 多个goroutine并发的累加计数器gs := 100wg.Add(gs)for i := 0; i < gs; i++ {go func() {defer wg.Done()// 累加for k := 0; k < 100; k++ {counter++// ++ 操作不是原子的// counter = counter + 1// 1. 获取当前的counter变量// 2. +1// 3. 赋值新值到counter}}()}// 统计计数结果wg.Wait()fmt.Println("Counter:", counter)}

Lock解决方案:

 func SyncLock() {n := 0wg := sync.WaitGroup{}​lk := sync.Mutex{}for i := 0; i < 1000; i++ {wg.Add(1)go func() {defer wg.Done()for i := 0; i < 100; i++ {lk.Lock()n++lk.Unlock()}}()}wg.Wait()fmt.Println("n:", n)}​// runn: 100000

互斥锁Mutex的使用

sync包提供了两种锁:

  • 互斥锁,Mutex

  • 读写互斥锁,RWMutex

互斥锁,同一时刻只能有一个goroutine申请锁定成功,不区分读、写操作。也称为:独占锁、排它锁。

提供了如下方法完成锁操作:

 type Mutex// 锁定锁m, 若锁m已是锁定状态,调用的goroutine会被阻塞,直到可以锁定func (m *Mutex) Lock()// 解锁锁m,若m不是锁定状态,会导致运行时错误func (m *Mutex) Unlock()// 尝试是否可以加锁,返回是否成功func (m *Mutex) TryLock() bool

注意:锁与goroutine没有关联,意味着允许一个goroutine加锁,在另一个goroutine中解锁。但是不是最典型的用法。

典型的锁用法:

 var lck sync.Mutexfunc () {lck.Lock()// 互斥执行的代码defer lck.Unlock()}

锁的流程:

image.png

示例:

 func SyncMutex() {wg := sync.WaitGroup{}var lck sync.Mutexfor i := 0; i < 4; i++ {wg.Add(1)go func(n int) {defer wg.Done()fmt.Println("before lock: ", n)lck.Lock()fmt.Println("locked: ", n)time.Sleep(1 * time.Second)lck.Unlock()fmt.Println("after lock: ", n)}(i)}wg.Wait()}

某次输出结果:

 before lock:  3locked:  3   before lock:  2before lock:  1before lock:  0after lock:  3locked:  2after lock:  2locked:  1after lock:  1locked:  0after lock:  0

可以发现,before lock 都是先执行的,而Lock() 操作,必须要等到其他goroutineUnlock()才能成功。

注意,如果其他goroutine没有通过相同的锁(1没用锁,2用了其他锁)去操作资源,那么是不受锁限制的,例如:

 func SyncLockAndNo() {n := 0wg := sync.WaitGroup{}lk := sync.Mutex{}for i := 0; i < 1000; i++ {wg.Add(1)go func() {defer wg.Done()for i := 0; i < 100; i++ {lk.Lock()n++lk.Unlock()}}()}wg.Add(1)go func() {defer wg.Done()for i := 0; i < 10000; i++ {n++}}()​// 其他锁//var lk2 sync.Mutex//go func() {//    defer wg.Done()//    for i := 0; i < 10000; i++ {//        lk2.Lock()//        n++//        lk2.Unlock()//    }//}()​wg.Wait()fmt.Println("n:", n)}​// 其中一次结果n: 109876

我们在第一个counter的例子上,增加了一个goroutine同去累加计数器counter,但没有使用前面的Mutex(不使用或使其他锁)。可见,出现了资源争用的情况。因此要注意:如果要限制资源的并发争用,要全部的资源操作都使用同一个锁。

实操时,锁除了直接调用外,还经常性出现在结构体中,以某个字段的形式出现,用于包含struct字段不会被多gorutine同时修改,例如我们 cancelCtx:

 type cancelCtx struct {Context​mu       sync.Mutex            // protects following fieldsdone     atomic.Value          // of chan struct{}, created lazily, closed by first cancel callchildren map[canceler]struct{} // set to nil by the first cancel callerr      error                 // set to non-nil by the first cancel call}

我们通常也会这么做,示例:

 type Post struct {Subject string// 赞Likes   int// 操作锁定mu sync.Mutex}​func (p *Post) IncrLikes() *Post {p.mu.Lock()defer p.mu.Unlock()p.Likes++​return p}​func (p *Post) DecrLikes() *Post {p.mu.Lock()defer p.mu.Unlock()p.Likes--​return p}

读写RWMutex的使用

读写互斥锁,将锁操作类型做了区分,分为读锁和写锁,由sync.RWMutex类型实现:

  • 读锁,Read Lock,共享读,阻塞写

  • 写锁,Lock,独占操作,阻塞读写

并发
支持不支持
不支持不支持

之所以减小锁的粒度,因为实际操作中读操作的比例要远高于写操作的比例,增加了共享读操作锁后,可以更大程度的提升读的并发能力。

sync.RWMutex 提供了如下方法完成操作:

 type RWMutex// 写锁定func (rw *RWMutex) Lock()// 写解锁func (rw *RWMutex) Unlock()​// 读锁定func (rw *RWMutex) RLock()// 读解锁func (rw *RWMutex) RUnlock()​// 尝试加写锁定func (rw *RWMutex) TryLock() bool// 尝试加读锁定func (rw *RWMutex) TryRLock() bool

写锁定,与互斥锁Mutex的语法和操作结果一致,都是保证互斥的独占操作。

读锁定,可以在已经存在读锁的情况下,加锁成功。

如图所示:

image.png

读锁示例:

 func SyncRLock() {wg := sync.WaitGroup{}// 模拟多个goroutinevar rwlck sync.RWMutexfor i := 0; i < 10; i++ {wg.Add(1)go func() {defer wg.Done()////rwlck.Lock()rwlck.RLock()// 输出一段内容fmt.Println(time.Now())time.Sleep(1 * time.Second)////rwlck.Unlock()rwlck.RUnlock()}()}​wg.Add(1)go func() {defer wg.Done()//rwlck.Lock()//rwlck.RLock()// 输出一段内容fmt.Println(time.Now(), "Lock")time.Sleep(1 * time.Second)//rwlck.Unlock()//rwlck.RUnlock()}()​wg.Wait()}

其中,使用读锁,输出操作会全部立即执行,然后集体sleep1s后全部结束。使用写锁,输出和Sleep会间隔1s依次执行。

实操示例:

 type Article struct {Subject string// 赞likes int// 操作锁定mu sync.RWMutex}​func (a Article) Likes() int {a.mu.RLock()defer a.mu.RUnlock()return a.likes}​func (a *Article) IncrLikes() *Article {a.mu.Lock()defer a.mu.Unlock()a.likes++return a}

同步Map sync.Map

Go中Map是非线程(goroutine)安全的。并发操作 Map 类型时,会导致 fatal error: concurrent map read and map write错误:

 func SyncMapErr() {m := map[string]int{}// 并发map写go func() {for {m["key"] = 0}}()// 并发map读go func() {for {_ = m["key"]}}()// 阻塞select {}}

之所以Go不支持Map的并发安全,是因为Go认为Map的典型使用场景不需要在多个Goroutine间并发安全操作Map。

并发安全操作Map的方案:

  • 锁 + Map,自定义Map操作,增加锁的控制,可以选择 Mutex和RWMutex。

  • sync.Map,sync包提供的安全Map.

锁+Map示例,在结构体内嵌入sync.Mutex:

 func SyncMapLock() {myMap := struct {sync.RWMutexData map[string]int}{Data: map[string]int{},}​// writemyMap.Lock()myMap.Data["key"] = 0myMap.Unlock()​// readmyMap.RLock()_ = myMap.Data["key"]myMap.RUnlock()}

sync.Map 的使用

 type Map// 最常用的4个方法:// 存储func (m *Map) Store(key, value any)// 遍历 mapfunc (m *Map) Range(f func(key, value any) bool)// 删除某个key元素func (m *Map) Delete(key any)// 返回key的值。存在key,返回value,true,不存在返回 nil, falsefunc (m *Map) Load(key any) (value any, ok bool)​// 若m[key]==old,执行删除。key不存在,返回falsefunc (m *Map) CompareAndDelete(key, old any) (deleted bool)// 若m[key]==old,执行交换, m[key] = newfunc (m *Map) CompareAndSwap(key, old, new any) bool​// 返回值后删除元素。loaded 表示是否load成功,key不存在,loaded为falsefunc (m *Map) LoadAndDelete(key any) (value any, loaded bool)// 加载,若加载失败则存储。返回加载或存储的值和是否加载func (m *Map) LoadOrStore(key, value any) (actual any, loaded bool)​// 存储新值,返回之前的值。loaded表示key是否存在func (m *Map) Swap(key, value any) (previous any, loaded bool)

sync.Map 不需要类型初始化,即可使用,可以理解为map[comparable]any。

使用示例,不会触发 fatal error: concurrent map read and map write

 func SyncSyncMap() {var m sync.Mapgo func() {for {m.Store("key", 0)}}()go func() {for {_, _ = m.Load("key")}}()select {}}

使用示例:

 func SyncSyncMapMethod() {wg := sync.WaitGroup{}var m sync.Mapfor i := 0; i < 10; i++ {wg.Add(1)go func(n int) {defer wg.Done()m.Store(n, fmt.Sprintf("value: %d", n))}(i)}​for i := 0; i < 10; i++ {wg.Add(1)go func(n int) {defer wg.Done()fmt.Println(m.Load(n))}(i)}​wg.Wait()m.Range(func(key, value any) bool {fmt.Println(key, value)return true})​//m.Delete(4)}

并发安全操作Map的方案的选择,统计的压测数据显示,相对而言:

  • 锁 + Map,写快,读慢

  • sync.Map,读快,写慢,删快,适合读多写少的场景

原子操作 sync/atomic

原子操作即是进行过程中不能被中断的操作,针对某个值的原子操作在被进行的过程中,CPU绝不会再去进行其他的针对该值的操作。为了实现这样的严谨性,原子操作仅会由一个独立的CPU指令代表和完成。原子操作是无锁的,常常直接通过CPU指令直接实现。 事实上,其它同步技术的实现常常依赖于原子操作。

原子操作是CPU指令级别实现的,比如在Intel的CPU上主要是使用总线锁的方式,AMD的CPU架构机器上就是使用MESI一致性协议的方式来保证原子操作。

go中 sync/atomic 包提供了原子操作的支持,用于同步操作整型(和指针类型):

  • int32

  • int64

  • uint32

  • uint64

  • uintptr

  • unsafe.Pointer

针对于以上类型,提供了如下操作:

 // Type 是以上的类型之一// 比较相等后交换 CASfunc CompareAndSwapType(addr *Type, old, new Type) (swapped bool)// 交换func SwapType(addr *Type, new Type) (old Type)// 累加func AddType(addr *Type, delta Type) (new Type)// 获取func LoadType(addr *Type) (val Type)// 存储func StoreType(addr *Type, val Type)

除了以上函数,还提供了对应的类型方法操作,以Int32为例:

 type Int32func (x *Int32) Add(delta int32) (new int32)func (x *Int32) CompareAndSwap(old, new int32) (swapped bool)func (x *Int32) Load() int32func (x *Int32) Store(val int32)func (x *Int32) Swap(new int32) (old int32)

除了以上几个整型,bool类型也提供了类型上的原子操作:

 type Boolfunc (x *Bool) CompareAndSwap(old, new bool) (swapped bool)func (x *Bool) Load() boolfunc (x *Bool) Store(val bool)func (x *Bool) Swap(new bool) (old bool)

示例:

 func SyncAtomicAdd() {// 并发的过程,没有加锁,Lock//var counter int32 = 0// type// atomic 原子的Int32, counter := 0counter := atomic.Int32{}wg := sync.WaitGroup{}for i := 0; i < 1000; i++ {wg.Add(1)go func() {defer wg.Done()for i := 0; i < 100; i++ {//atomic.AddInt32(&counter, 1)// type// 原子累加操作 , counter ++counter.Add(1)}}()}wg.Wait()//fmt.Println("counter:", atomic.LoadInt32(&counter))// typefmt.Println("counter:", counter.Load())}

以上示例不会出现不到10000的情况了。

除了预定义的整型的支持,还可以使用 atomic.Value 类型,完成其他类型的原子操作:

 type Valuefunc (v *Value) CompareAndSwap(old, new any) (swapped bool)func (v *Value) Load() (val any)func (v *Value) Store(val any)func (v *Value) Swap(new any) (old any)

使用方法:

 func SyncAtomicValue() {​var loadConfig = func() map[string]string {return map[string]string{// some config"title":   "马士兵Go并发编程","varConf": fmt.Sprintf("%d", rand.Int63()),}}​var config atomic.Value​// 每N秒加载一次配置文件go func() {for {config.Store(loadConfig())fmt.Println("latest config was loaded", time.Now().Format("15:04:05.99999999"))time.Sleep(time.Second)}}()​// 使用配置// 不能在加载的过程中使用配置for {go func() {c := config.Load()fmt.Println(c, time.Now().Format("15:04:05.99999999"))}()​time.Sleep(400 * time.Millisecond)}​select {}}

sync.Pool 并发安全池

image.png

池是一组可以单独保存和检索的可以复用的临时对象。存储在池中的任何项目可随时自动删除,无需通知。一个池可以安全地同时被多个goroutine使用。

典型特征:

  • sync.Pool 是并发安全的

  • 池中的对象由Go负责删除,内存由Go自己回收

  • 池中元素的数量由Go负责管理,用户无法干预

  • 池中元素应该是临时的,不应该是持久的。例如长连接不适合放入 sync.Pool 中

池的目的是缓存已分配但未使用的项目以供以后重用,从而减轻垃圾收集器的压力。也就是说,它使构建高效、线程安全的自由元素变得容易。

池的一个适当用途是管理一组临时项,这些临时项在包的并发独立客户端之间默默共享,并可能被其重用。池提供了一种在许多客户机上分摊分配开销的方法。

一个很好地使用池的例子是fmt包,它维护了临时输出缓冲区的动态大小存储。

池由 sync.Pool类型实现,具体三个操作:

  • 初始化Pool实例,需要提供池中缓存元素的New方法。

  • 申请元素,func (p *Pool) Get() any

  • 交回对象,func (p *Pool) Put(x any)

操作示例:

 func SyncPool() {// 原子的计数器var counter int32 = 0​// 定义元素的Newer,创建器elementNewer := func() any {// 原子的计数器累加atomic.AddInt32(&counter, 1)​// 池中元素推荐(强烈)是指针类型return new(bytes.Buffer)}​// Pool的初始化pool := sync.Pool{New: elementNewer,}​// 并发的申请和交回元素workerNum := 1024 * 1024wg := sync.WaitGroup{}wg.Add(workerNum)for i := 0; i < workerNum; i++ {go func() {defer wg.Done()// 申请元素,通常需要断言为特定类型buffer := pool.Get().(*bytes.Buffer)// 不用Pool//buffer := elementNewer().(*bytes.Buffer)// 交回元素defer pool.Put(buffer)// 使用元素_ = buffer.String()}()}​//wg.Wait()​// 测试创建元素的次数fmt.Println("elements number is :", counter)}​// elements number is : 12

测试的时候,大家可以发现创建的元素数量远远低于goroutine的数量。

DATA RACE 现象

当程序运行时,由于并发的原因会导致数据竞争使用,有时在编写代码时很难发现,要经过大量测试才会发现。可以用 go run -race ,增加-race选项,检测运行时可能出现的竞争问题。

测试之前的计数器累加代码:本例子需要 main.main 来演示,因为是 go run:

package mainimport ("fmt""sync"
)func main() {wg := sync.WaitGroup{}// 计数器counter := 0// 多个goroutine并发的累加计数器gs := 1000wg.Add(gs)for i := 0; i < gs; i++ {go func() {defer wg.Done()// 累加for k := 0; k < 100; k++ {counter++// ++ 操作不是原子的// counter = counter + 1// 1. 获取当前的counter变量// 2. +1// 3. 赋值新值到counter}}()}// 统计计数结果wg.Wait()fmt.Println("Counter:", counter)
}

结果:

# 没有使用 -race
PS D:\apps\goExample\concurrency> go run .\syncRace.go
n: 94077# 使用 -race
PS D:\apps\goExample\concurrency> go run -race .\syncRace.go
==================
WARNING: DATA RACE  
Read at 0x00c00000e0f8 by goroutine 9:main.main.func1()D:/apps/goExample/concurrency/syncMain.go:16 +0xa8Previous write at 0x00c00000e0f8 by goroutine 7:
Goroutine 9 (running) created at:main.main()D:/apps/goExample/concurrency/syncMain.go:13 +0x84Goroutine 7 (finished) created at:main.main()D:/apps/goExample/concurrency/syncMain.go:13 +0x84
==================
n: 98807
Found 1 data race(s)
exit status 66

该选项用于在开发阶段,检测数据竞争情况。

出现 data race情况,可以使用锁,或原子操作的来解决。

sync.Once

若需要保证多个并发goroutine中,某段代码仅仅执行一次,就可以使用 sync.Once 结构实现。

例如,在获取配置的时候,往往仅仅需要获取一次,然后去使用。在多个goroutine并发时,要保证能够获取到配置,同时仅获取一次配置,就可以使用sync.Once结构:

func SyncOnce() {// 初始化config变量config := make(map[string]string)// 1. 初始化 sync.Onceonce := sync.Once{}// 加载配置的函数loadConfig := func() {// 2. 利用 once.Do() 来执行once.Do(func() {// 保证执行一次config = map[string]string{"varInt": fmt.Sprintf("%d", rand.Int31()),}fmt.Println("config loaded")})}// 模拟多个goroutine,多次调用加载配置// 测试加载配置操作,执行了几次workers := 10wg := sync.WaitGroup{}wg.Add(workers)for i := 0; i < workers; i++ {go func() {defer wg.Done()// 并发的多次加载配置loadConfig()// 使用配置_ = config}()}wg.Wait()
}

核心逻辑:

  1. 初始化 sync.Once

  2. once.Do(func()) 可以确保func()仅仅执行一次

sync.Once 的实现很简单:

type Once struct {// 是否已处理,保证一次done uint32// 锁,保证并发安全m    Mutex
}

sync.Cond

sync.Cond是sync包提供的基于条件(Condition)的通知结构。

该结构提供了4个方法:

// 创建Cond
func NewCond(l Locker) *Cond
// 全部唤醒
func (c *Cond) Broadcast()
// 唤醒1个
func (c *Cond) Signal()
// 等待唤醒
func (c *Cond) Wait()

其中,创建时,需要1个Locker作为参数,通常是 sync.Mutext或sync.RWMutex。然后两个方法用来通知,一个方法用来等待。

使用逻辑很简单,通常是一个goroutine负责通知,多个goroutine等待处理,如图:

image.png

创建Cond,sync.NewCond() 需要提供锁,同时在等待操作和广播(信号)操作中,通常需要先申请锁,其中等待操作是必须的,而官博(信号)操作是可选的。,例如:

cond := sync.NewCond(&sync.Mutex{})
cond := sync.NewCond(&sync.RWMutex{})

还有,cond的广播和信号通知操作是并发安全的,可以重复调用的。

要注意Wait()操作,是会先解锁,等到广播信号后,再加锁。因此,Wait()操作前,要加锁。

示例代码:

  • 一个goroutine负责接收数据,完毕后,广播给处理数据的goroutine

  • 多个goroutine处理数据,在数据未处理完前,等待广播信号。信号来了,处理数据

func SyncCond() {wg := sync.WaitGroup{}dataCap := 1024 * 1024var data []intcond := sync.NewCond(&sync.Mutex{})for i := 0; i < 8; i++ {wg.Add(1)go func(c *sync.Cond) {defer wg.Done()c.L.Lock()for len(data) < dataCap {c.Wait()}fmt.Println("listen", len(data), time.Now())c.L.Unlock()}(cond)}wg.Add(1)go func(c *sync.Cond) {defer wg.Done()c.L.Lock()defer c.L.Unlock()for i := 0; i < dataCap; i++ {data = append(data, i*i)}fmt.Println("Broadcast")c.Broadcast()//c.Signal()}(cond)// 为什么 for { wait() }// 另外的广播goroutine//wg.Add(1)//go func(c *sync.Cond) {//    defer wg.Done()//    c.Broadcast()//}(cond)wg.Wait()
}

示例代码要点:

  • wait所在的goroutine要判定是否需要wait,所以wait要出现在条件中,因为goroutine调用的关系,不能保证wait在broadcast前面执行

  • wait要使用for进行条件判定,因为在wait返回后,条件不一定成立。因为Broadcast()操作可能被提前调用(通常是在其他的goroutine中。

  • Broadcast() 操作可选的是否加锁解锁

  • Wait() 操作前,一定要加锁。因为Wait()操作,会先解锁,接收到信号后,再加锁。

sync.Cond 基本原理

sync.Cond结构:

type Cond struct {// 锁L Locker// 等待通知goroutine列表notify  notifyList// 限制不能被拷贝noCopy noCopychecker copyChecker
}

结构上可见,Cond记录了等待的goroutine列表,这样就可以做到,广播到全部的等待goroutine。这也是Cond应该被复制的原因,否则这些goroutine可能会被意外唤醒。

Wait() 操作:

func (c *Cond) Wait() {// 检查是否被复制c.checker.check()// 更新 notifyList 中需要等待的 waiter 的数量// 返回当前需要插入 notifyList 的编号t := runtime_notifyListAdd(&c.notify)// 解锁c.L.Unlock()// 挂起,直到被唤醒runtime_notifyListWait(&c.notify, t)// 唤醒之后,重新加锁。// 因为阻塞之前解锁了。c.L.Lock()
}

核心工作就是,记录当goroutine到Cond的notifyList。之后解锁,挂起,加锁。因此要在Wait()前加锁,后边要解锁。

Broadcast()操作:

func (c *Cond) Broadcast() {// 检查 sync.Cond 是否被复制了c.checker.check()// 唤醒 notifyList 中的所有 goroutineruntime_notifyListNotifyAll(&c.notify)
}

核心工作就是唤醒 notifyList 中全部的 goroutine。

小结

同步类型:

  • 数据同步,保证数据操作的原子性

    • sync/atomic

    • sync.Map

    • sync.Mutex, sync.RWMutex

  • 操作同步

    • sync.Mutex, sync.RWMutex

锁的类型:

  • 互斥锁 sync.Mutex,完全独占

  • 读写互斥锁 sync.RWMutex,可以共享读操作

锁的不锁资源,只是锁定申请锁本身的操作。

sync包总结

  • 锁:sync.Mutex, sync.RWMutex

  • 数据:sync.Map, sync/atomic

  • sync.Pool

  • sync.Once

  • sync.Cond

使用Channel完成数据和信号的同步!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/3249105.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

P3-AI产品经理-九五小庞

AI产品的数据流向 美团外卖&#xff0c;实时只能调度 美团28分钟送达需求的分析 AI产品常用的算法 常用算法 常见的AI算法解析 自然语言生成NLG语音识别&#xff1a;科大讯飞&#xff0c;通义千问 虚拟现实机器学习平台 决策管理系统生物特征识别技术 RPA(机器人流程自动…

ICE-BA原理

文章目录 主要思想约束建立IMU&Local & globalBAIMU预积分LocalBAGlobalBA求解方法常用的求解思路改进增量BA方法 论文&#xff1a; 《ICE-BA: Incremental, Consistent and Efficient Bundle Adjustment for Visual-Inertial SLAM》 ICE-BA&#xff1a;增量、一致且高…

多商户SaaS版扫码点餐系统开源

基于前后端分离的 多商户 SaaS 版扫码点餐系统&#xff0c;支持后台点餐、多人同时在线点餐、购物车共享、餐桌状态实时监控&#xff0c;菜品管理、餐桌管理等众多功能。 源码下载&#xff1a;多商户 SaaS 版扫码点餐系统.zip 功能特色 手机扫码点餐功能&#xff1a;用户可…

如何评价2023年国际高校数学建模竞赛B题?

问题1&#xff1a;在三星堆发现的一个外圆直径为85厘米&#xff0c;内圆直径为40厘米&#xff08;假设&#xff09;的青铜车轮&#xff0c;青铜车轮有5条车轴&#xff08;射线&#xff09;&#xff0c;5条内弧线本质是双曲线&#xff0c;顶点与内圆相切&#xff0c;内弧双曲线的…

植物病害分级调研

Web of Science搜索&#xff0c;关键字“plant disease severity recognition”&#xff0c;共407篇&#xff0c;限制2023年以后共71篇 2019、2020 《Disentangled Representation Learning for Leaf Diseases Recognition》 2019 IF&#xff1a;0.8 论文&#xff1a;Disen…

Xcode进行真机测试时总是断连,如何解决?

嗨。大家好&#xff0c;我是兰若姐姐。最近我在用真机进行app自动化测试的时候&#xff0c;经常会遇到xcode和手机断连&#xff0c;每次断连之后需要重新连接&#xff0c;每次断开都会出现以下截图的报错 当这种情况出现时&#xff0c;之前执行的用例就相当于白执行了&#xff…

爬虫瑞数5案例:某大学总医院

声明: 该文章为学习使用,严禁用于商业用途和非法用途,违者后果自负,由此产生的一切后果均与作者无关 一、瑞数简介 瑞数动态安全 Botgate(机器人防火墙)以“动态安全”技术为核心,通过动态封装、动态验证、动态混淆、动态令牌等技术对服务器网页底层代码持续动态变换,…

Go语言并发编程-Goroutine调度

goroutine 概念 在Go中&#xff0c;每个并发执行的单元称为goroutine。通常称为Go协程。 go 关键字启动goroutine go中使用关键字 go 即可启动新的goroutine。 示例代码&#xff1a; 两个函数分别输出奇数和偶数。采用常规调用顺序执行&#xff0c;和采用go并发调用&…

大模型学习笔记十一:视觉大模型

一、判别式模型和生成式模型 1&#xff09;判别式模型Discriminative ①给某一个样本&#xff0c;判断属于某个类别的概率&#xff0c;擅长分类任务&#xff0c;计算量少。&#xff08;学习策略函数Y f(X)或者条件概率P(YIX)&#xff09; ②不能反映训练数据本身的特性 ③学习…

优思学院|直方图与条形图的具体区别

在六西格玛方法、质量管理工具中&#xff0c;数据的分析和可视化是关键步骤。直方图和条形图是两种常用的图表工具&#xff0c;但它们在用途和显示方式上有显著区别。本文将详细探讨这两种图表的定义、特性、应用及如何选择适合的图表。 1. 直方图和条形图的定义 直方图是一种…

人工智能未来发展前景将会怎样?

当我们探讨人工智能未来的发展前景时&#xff0c;可以从多个角度来详细说明其可能的影响和趋势&#xff1a; 技术进步与应用扩展 1.深度学习与机器学习&#xff1a; 进一步优化和算法进展&#xff1a;深度学习已经取得了巨大成就&#xff0c;但仍面临挑战&#xff0c;如对小数…

程序员想要6万一个月,需要什么能力,要吃什么样的苦?

让我们来算一道小学数学题&#xff1a;6w*1272w&#xff0c;年包72w的程序员起码是阿里P7-P8的水平了&#xff0c;论工作职责来说&#xff0c;起码得是大厂的一个小tech leader&#xff0c;如果是在小公司&#xff0c;基本上是公司骨干级成员&#xff0c;或是统筹整个项目和小组…

FFmpeg播放视频

VS2017+FFmpeg6.2.r113110+SDL2.30.5 1.下载 ShiftMediaProject/FFmpeg 2.下载SDL2 3.新建VC++控制台应用 3.配置include和lib 4.把FFmpeg和SDL的dll 复制到工程Debug目录下,并设置调试命令

如何让您的反爬虫策略更具弹性?揭秘管理技巧

摘要&#xff1a; 本文深入探讨了反爬虫策略的最新趋势与实战技巧&#xff0c;旨在帮助网站所有者和数据分析师构建更加灵活高效的爬虫管理系统。通过理解反爬机制、动态应对策略及合法数据采集的最佳实践&#xff0c;确保数据收集在遵守网络规则的同时&#xff0c;实现业务目…

Kettle 登录示例 POST请求

登录接口是post请求&#xff0c;组装Body为json字符串 var body "{\"username\":\""username"\",\"password\": \""password"\",\"code\":\""verification"\",\"uuid\…

YOLOv7网络结构学习

YOLOV7详细解读&#xff08;一&#xff09;网络架构解读 YOLOV7学习记录之原理代码介绍 【Make YOLO Great Again】YOLOv1-v7全系列大解析&#xff08;Backbone篇&#xff09; yolov7 图解 深入浅出 Yolo 系列之 Yolov7 基础网络结构详解 我觉得Head、Neck和Head的划分不太…

FedAvg的简单实现(详解)

对于联邦学习正在学习中&#xff0c;下文中若有错误出现&#xff0c;望指正 介绍 本文在简单实现联邦平均算法时&#xff0c;使用客户-服务器架构&#xff0c;其基本流程是&#xff1a; 1、server初始化模型参数&#xff0c;所有clients将这个初始模型下载到本地 2、clien…

每个人都有良知,只是被遮蔽的程度不同

85天 【困之勉行&#xff0c;下笨功夫】 每个人的良知余光都在&#xff0c;困之勉行努力用余光去精细明察&#xff0c;须下“人一己百&#xff0c;人十己千”的努力&#xff1b; 生活中&#xff0c;我们往往会看到&#xff0c;绝顶聪明的人往往愿意下笨功夫&#xff0c;而资质…

linux开机后不用登陆,无法正常进入系统,出现:/#的提示符

linux开机后不用登陆&#xff0c;无法正常进入系统&#xff0c;出现:/#的提示符 解决方案&#xff1a; 1、输入命令 ls /dev/mapper 此时会出现3个文件。其中rhel-root文件 是我们下面所要用的文件。 ls的目的就是为了让大家能知道自己带"-root" 文件的前缀是什…

C语言switch的使用

switch的使用语句 switch&#xff08;表达式&#xff09; { case 值1&#xff1a; 语句1; break; case 值2&#xff1a; 语句2; break; default: break; } 注意事项&#xff1a;1.表达式计算结果只能为&#xff08;字符/整数&#xff09; 2.case值只能是&#xff08;字…