我正在考虑按照协议构建UDP数据包,该协议将数据包的布局指定为位级别,我不确定如何在Ruby中进行此操作。我的理解是数组的pack
和unpack
函数是最合适的。
http://ruby-doc.org/core-2.2.0/Array.html#method-i-pack
数组文档列出了一大堆可以与pack
一起使用的不同参数,但我不确定如何将一个布尔值打包为一位。
我试图实现的格式看起来像那样。
我的理解是我可以做类似的事情:
[size_int, origin_as_int, tagged, addressable, protocol_int, source_int].pack "v1????V1"
问题标记表示我不知道如何表示字段。我可以做类似
的事情binary_string = ""
binary_string += "%02b" % origin_as_int
binary_string += (tagged ? "1" : "0")
binary_string += (addressable ? "1" : "0")
binary_string += "%012b" % protocol_int
munged_stuff = binary_string.to_i(2)
[size_int, munged_stuff, source_int].pack "v2V1"
我想?但是,感觉很糟糕。
答案 0 :(得分:2)
我从未使用引发头痛的方法Array#pack和String#unpack,但在阅读完他们的文档之后,它们实际上看起来非常简单,至少我认为这是更常用的指令,例如'C'
,'S'
,'L'
,'Q'
,'B'
,'b'
,'A'
和'a'
。
我相信你可以这样做:
size_int = 123 # C (0 to 2**8-1 => 255)
origin_as_int = "10" # B2
tagged = "1" # B1
addressable = "0" # B1
protocol_int = "010011001011" # B12
source_int = 2361 # S (0 to 2**16-1 => 65535)
template = %| C B2 B1 B1 B12 S |.delete(' ')
#=> "CB2B1B1B12S"
s = [size_int, origin_as_int, tagged, addressable, protocol_int,
source_int].pack(template)
#=> "{\x80\x80\x00L\xB09\t"
s.unpack(template)
#=> [123, "10", "1", "0", "010011001011", 2361]