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

做照片的网站有哪些软件网络推广专员所需知识

做照片的网站有哪些软件,网络推广专员所需知识,餐饮系统网站建设,广西建网站公司并发访问slice 线上出现一粒多协程并发append全局slice的情况,导致内存不断翻倍,因此对slice的使用需要重新考虑。 并发读写的情况下, 可以利用锁、channel等避免竞态 问题 func TestDemo32(t *testing.T) {var wg sync.WaitGroupvar n 1…

并发访问slice

线上出现一粒多协程并发append全局slice的情况,导致内存不断翻倍,因此对slice的使用需要重新考虑。
并发读写的情况下, 可以利用锁、channel等避免竞态

问题

func TestDemo32(t *testing.T) {var wg sync.WaitGroupvar n = 100s := make([]int, 0, 200)hdr := (*reflect.SliceHeader)(unsafe.Pointer(&s))fmt.Printf("Data addr: %d\n", hdr.Data)fmt.Printf("Data cap: %d\n", hdr.Cap)fmt.Printf("Data len: %d\n", hdr.Len)wg.Add(n)for i := 1; i <= n; i++ {go func(v int) {defer wg.Done()s = append(s, v)}(i)}wg.Wait()hdr = (*reflect.SliceHeader)(unsafe.Pointer(&s))fmt.Printf("Data addr: %d\n", hdr.Data)fmt.Printf("Data cap: %d\n", hdr.Cap)fmt.Printf("Data len: %d\n", hdr.Len)fmt.Println(jsonx.ToString(s))// Data addr: 824645965056// Data cap: 200// Data len: 0// Data addr: 824645965056// Data cap: 200// Data len: 96
}func TestDemo33(t *testing.T) {var wg sync.WaitGroupvar n = 500s := make([]int, 0, 10)hdr := (*reflect.SliceHeader)(unsafe.Pointer(&s))fmt.Printf("Data addr: %d\n", hdr.Data)fmt.Printf("Data cap: %d\n", hdr.Cap)fmt.Printf("Data len: %d\n", hdr.Len)wg.Add(n)for i := 1; i <= n; i++ {go func(v int) {defer wg.Done()s = append(s, v)}(i)}wg.Wait()hdr = (*reflect.SliceHeader)(unsafe.Pointer(&s))fmt.Printf("Data addr: %d\n", hdr.Data)fmt.Printf("Data cap: %d\n", hdr.Cap)fmt.Printf("Data len: %d\n", hdr.Len)fmt.Println(jsonx.ToString(s))// Data addr: 824635459136// Data cap: 10// Data len: 0// Data addr: 824665328128// Data cap: 672// Data len: 453
}

Go语言中的slice是一种引用类型,它本身不保存任何元素,只是对数组的引用。
append操作的主要功能是向slice添加元素。
如果slice的底层数组容量足够,就直接添加;如果不够,就会先分配新的底层数组,然后将原有的元素和新添加的元素一起拷贝到新数组,在这个过程中,原有的底层数组会被垃圾回收。

当底层数组的容量不足以容纳新的元素时,就会产生新的底层数组
此时原有的slice和返回的新的slice,其底层数组是不同的。
这也是为什么在使用append时,总是习惯性地将结果再次赋值给原slice。

方案一 channel

func TestDemo(t *testing.T) {// 无缓冲,发送侧有数据,接收侧才执行// 用于做同步c := make(chan struct{})// new 了该 job 后,该 job 就开始准备从 channel 接收数据s := NewScheduleJob(n, func() { c <- struct{}{} })// 并发发送数据到channelvar wg sync.WaitGroupvar n  = 1000wg.Add(n)for i := 0; i < n; i++ {go func(v int) {defer wg.Done()s.AddData(v)}(i)}// 等待上述多个协程将数据存入slicewg.Wait()// 发送完之后关闭channels.Close()// 阻塞在这里是等待NewScheduleJob执行结束<-c// 最终实现读写一致fmt.Println(len(s.data))
}type ServiceData struct {ch   chan int // 用来同步的channeldata []int    // 存储数据的slice
}// Schedule 从 channel 接收数据串行存入slice,直到ch关闭
func (s *ServiceData) Schedule() {for i := range s.ch {s.data = append(s.data, i)}
}// Close 关闭channel
func (s *ServiceData) Close() {close(s.ch)
}// AddData 发送数据到 channel
func (s *ServiceData) AddData(v int) {s.ch <- v
}func NewScheduleJob(size int, done func()) *ServiceData {s := &ServiceData{ch:   make(chan int, size),data: make([]int, 0),}go func() {// 并发地 append 数据到 slice// Schedule 从 channel 接收数据串行存入slice,直到ch关闭s.Schedule()// 通知主协程继续执行done()}()return s
}

优点:逻辑复杂,适用于高并发场景

方案二 - 锁

func main() {slc := make([]int, 0, 1000)var wg sync.WaitGroupvar lock sync.Mutexfor i := 0; i < 1000; i++ {wg.Add(1)go func(a int) {defer wg.Done()lock.Lock()defer lock.Unlock()slc = append(slc, a)}(i)}wg.Wait()fmt.Println(len(slc))
}

优点:锁的逻辑重,适用于对性能要求不高的场景

学习文档

https://juejin.cn/post/6844904134592692231

http://www.yidumall.com/news/59990.html

相关文章:

  • 小企业网站建设怎么做好北京首页关键词优化
  • 学做网站用什么软件b站推广网站入口2023的推广形式
  • 网站开发相关会议做外贸怎么推广
  • 免费建设网站网站设计制作在哪能看
  • 哪个网站可以做空比特币网络产品运营与推广
  • 优秀企业网站案例自媒体135的网站是多少
  • 九里网站开发哈尔滨seo优化软件
  • 网站建设是 口号福州seo推广公司
  • 四川网站建设博客网站统计分析平台
  • 徐州哪家做网站好智推教育seo课程
  • 福州网站建设流程免费建自己的网站
  • 网站建设与网页设计制作书籍信阳网站推广公司
  • 南京网站设计公司兴田德润电话多少长沙网站seo诊断
  • 外贸 网站 源码天眼查企业查询入口
  • 兼职做彩平网站网站排名优化公司
  • 洛阳建站公司贵州seo推广
  • 武汉市大型的网站制作公司营销方式都有哪些
  • 发广告去哪个平台北京seo优化诊断
  • 网站开发 相册竞价推广代运营服务
  • wordpress数据库恢复插件苏州seo推广
  • 网站链接怎么做二维码国际新闻最新消息十条
  • 卢湾网站东莞网站推广运营公司
  • 大讲堂123专注网站模板制作赛雷猴是什么意思
  • 海南私彩网站开发找关键词的方法与技巧
  • 企业网站平台如何做网络推广网络销售培训
  • 在哪个网站可以做酒店预定单企业查询网站
  • 网络app开发网站建设价格媒体发稿费用
  • 做网站的主题有哪些怎么推广自己的微信号
  • wordpress临时关闭站点广告主资源哪里找
  • 关于做网站的书籍制作网页一般多少钱