type EventPrefs struct {
Call bool
Presence bool
Endpoint bool
VoiceMail bool
CallRecording bool
}
目前,该结构类型的大小为 5 个字节,但我想使用位。有没有办法做到这一点?
type EventPrefs struct {
Call bool
Presence bool
Endpoint bool
VoiceMail bool
CallRecording bool
}
目前,该结构类型的大小为 5 个字节,但我想使用位。有没有办法做到这一点?
Go 中没有“位”类型,因此如果要将多个bool
信息打包成位,则必须自己实现。声明一个类型uint8
(或uint16
任何其他整数类型)的字段,并提供获取/设置该字段特定位的方法。
一般位设置/清除就像这样简单:
var masks = []uint8{0x01, 0x02, 0x04, 0x08, 0x10}
func set(field, data uint8, b bool) uint8 {
if b {
return data | masks[field] // Set bit
}
return data ^ masks[field] // Clear bit
}
func get(field, data uint8) bool {
return data&masks[field] != 0
}
将 5 个布尔字段打包成一个uint8
值:
type EventPrefs struct {
data uint8
}
func (e *EventPrefs) SetCall(b bool) { e.data = set(0, e.data, b) }
func (e *EventPrefs) Call() bool { return get(0, e.data) }
func (e *EventPrefs) SetPresence(b bool) { e.data = set(1, e.data, b) }
func (e *EventPrefs) Presence() bool { return get(1, e.data) }
func (e *EventPrefs) SetEndpoint(b bool) { e.data = set(2, e.data, b) }
func (e *EventPrefs) Endpoint() bool { return get(2, e.data) }
func (e *EventPrefs) SetVoiceMail(b bool) { e.data = set(3, e.data, b) }
func (e *EventPrefs) VoiceMail() bool { return get(3, e.data) }
func (e *EventPrefs) SetCallRecording(b bool) { e.data = set(4, e.data, b) }
func (e *EventPrefs) CallRecording() bool { return get(4, e.data) }
测试它:
ep := &EventPrefs{}
fmt.Println("Calls:", ep.Call(), ep.data)
ep.SetCall(true)
fmt.Println("Calls:", ep.Call(), ep.data)
fmt.Println("Presence:", ep.Presence(), ep.data)
ep.SetPresence(true)
fmt.Println("Presence:", ep.Presence(), ep.data)
ep.SetPresence(false)
fmt.Println("Presence:", ep.Presence(), ep.data)
哪些输出(在Go Playground上尝试):
Calls: false 0
Calls: true 1
Presence: false 1
Presence: true 3
Presence: false 1
节省 4 个字节值得吗?很少。
注意:上述解决方案可以有很多变化。例如,掩码可以使用位移“计算”,set()
andget()
函数可以是方法,EventPrefs
因此data
不需要参数(并且set()
可以直接设置EventPrefs.data
字段,因此也不需要返回值)。如果set()
仍然是一个函数,则data
参数可以是一个指针,因此set()
可以更改指向的值而不返回新的data
等。该data
字段可能有自己的声明类型,例如附加到它bitpack
的方法get()
和方法。set()