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

go语言中空结构体

在这里插入图片描述

空结构体(struct{})

  • 普通理解

在结构体中,可以包裹一系列与对象相关的属性,但若该对象没有属性呢?那它就是一个空结构体。

空结构体,和正常的结构体一样,可以接收方法函数。

type Lamp struct{}

func (l Lamp) On() {
        println("On")

}
func (l Lamp) Off() {
        println("Off")
}
  • 空结构体的妙用

空结构体的表象特征,就是没有任何属性,而从更深层次的角度来说,空结构体是一个不占用空间的对象。

使用 unsafe.Sizeof 可以轻易的验证这个结果

type Lamp struct{}

func main() {
    lamp := Lamp{}
    fmt.Print(unsafe.Sizeof(lamp))
}
// output: 0

基于这个特性,在一些特殊的场合之下,可以用做占位符使用,合理的使用空结构体,会减小程序的内存占用空间。

比如在使用信道(channel)控制并发时,我们只是需要一个信号,但并不需要传递值,这个时候,也可以使用 struct{} 代替。

func main() {
    ch := make(chan struct{}, 1)
    go func() {
        <-ch
        // do something
    }()
    ch <- struct{}{}
    // ...
}

在 Go 语言中,使用空结构体(struct{})作为通道(chan)的元素类型是一种常见的优化手段。这种做法主要出于以下几个原因:

  1. 节省内存
    空结构体 struct{} 在 Go 中不占用任何内存空间(大小为 0 字节)。因此,当你需要一个通道来传递信号或同步协程时,使用空结构体可以避免不必要的内存开销。

  2. 信号传递
    在某些场景下,你并不需要通过通道传递具体的数据,而只是需要一个简单的信号机制来通知其他协程某个事件已经发生。例如,用于关闭多个工作协程、通知某个操作完成等。此时,空结构体作为通道的元素类型非常合适。

  3. 提高性能
    由于空结构体不占用内存,发送和接收空结构体的操作通常比发送和接收复杂数据类型的通道更快。虽然这种差异在大多数情况下是微不足道的,但在高并发或高性能要求的场景下,这些细微的优化可能会产生显著的影响。

.关闭多个工作协程

package main

import (
    "fmt"
    "time"
)

func worker(id int, done chan struct{}) {
    for {
        select {
        case <-done:
            fmt.Printf("Worker %d shutting down\n", id)
            return
        default:
            fmt.Printf("Worker %d working\n", id)
            time.Sleep(500 * time.Millisecond)
        }
    }
}

func main() {
    done := make(chan struct{})
    numWorkers := 3

    // 启动多个工作协程
    for i := 1; i <= numWorkers; i++ {
        go worker(i, done)
    }

    // 模拟一些工作
    time.Sleep(2 * time.Second)

    // 发送关闭信号
    close(done)

    // 等待一段时间以确保所有工作协程都已退出
    time.Sleep(1 * time.Second)
}

在这个例子中,done 通道被用来通知所有工作协程停止工作。我们不需要通过通道传递任何实际的数据,只需要一个信号即可。

.同步操作完成

package main

import (
    "fmt"
    "sync"
)

func task(id int, wg *sync.WaitGroup, done chan struct{}) {
    defer wg.Done()
    fmt.Printf("Task %d completed\n", id)
    done <- struct{}{} // 发送一个空结构体表示任务完成
}

func main() {
    var wg sync.WaitGroup
    done := make(chan struct{}, 3) // 缓冲区大小为任务数量

    for i := 1; i <= 3; i++ {
        wg.Add(1)
        go task(i, &wg, done)
    }

    // 等待所有任务完成
    go func() {
        wg.Wait()
        close(done)
    }()

    // 接收所有完成信号
    for range done {
        fmt.Println("Received completion signal")
    }

    fmt.Println("All tasks completed")
}

在这个例子中,每个任务完成后都会向 done 通道发送一个空结构体,表示任务已完成。主协程通过读取 done 通道中的信号来确认所有任务是否已完成。

相关文章:

  • [代码规范]1_良好的命名规范能减轻工作负担
  • golang+redis 实现分布式限流
  • 蓝桥杯 握手问题
  • 【C#高阶编程】—单例模式详解
  • MySQL性能优化,sql优化有哪些,数据库如何优化设计(二)
  • 【软件工程】08_结构化设计方法
  • Bash 脚本基础
  • numpy学习笔记15:模拟100次随机游走,观察平均行为
  • 数据处理专题(二)
  • vue2 el-table跨分页多选以及多选回显
  • Springboot的MultipartFile,获取不到inputStream
  • SeaCMS代码审计
  • 基于深度学习的OCR+NLP,医疗化验单智能识别方案
  • 【量化实战】利用miniqmt实现远程下单的完整指南
  • 阿里开源QwQ-32B推理模型!32.5B vs 671B|仅需1/10成本
  • python函数的多种参数使用形式
  • R语言软件配置(自用)
  • 人工智能之数学基础:矩阵的降维
  • 对上传的图片进行压缩,以保证它的大小不超过X MB
  • 亚马逊新品广告投放策略:从零到爆单的全链路解析
  • 浦发银行一季度净利175.98亿增1.02%,不良率微降
  • 五月院线片单:就看五一档表现了
  • 日韩 “打头阵”与美国贸易谈判,汽车、半导体产业忧虑重重
  • 外交部:印度香客赴中国西藏神山圣湖朝圣将于今年夏季恢复
  • 贸促会答澎湃:5月22日将举办2025年贸易投资促进峰会
  • 央行副行长:增强外汇市场韧性,坚决对市场顺周期行为进行纠偏