字节

时间:2018-02-22 11:29:52

标签: scala literals scala-2.12

默认情况下,包含A-F数字的十六进制文字将转换为int。当我尝试使用 0x 来解析Int时,它正在正确创建。

val a: Int = 0x34
val b: Int = 0xFF

但是当我试图用 0x 去除字节时,第二行没有编译

val a: Byte = 0x34
val b: Byte = 0xFF // compilation error

我找到了一个

的解决方法
val a: Byte = 0x34
val b: Byte = 0xFF.toByte

但有没有体面的方法从其十六进制文字中删除一个Byte?

例如,我试图以这种方式在Test方法中去掉一个Byte数组

anObject.someMethod(1, 1.1f, 0xAB, "1") shouldBe
  Array[Byte](0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF, 0xAF)
anObject.someMethod(2, 2.2f, 0xCD, "2") shouldBe
  Array[Byte](0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE, 0xBE)
anObject.someMethod(3, 3.2f, 0xEF, "3") shouldBe
  Array[Byte](0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD, 0xCD)

但不是这样

anObject.someMethod(1, 1.1f, 0xAB.toByte, "1") shouldBe
  Array[Byte](0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte, 0xAF.toByte)
anObject.someMethod(2, 2.2f, 0xCD.toByte, "2") shouldBe
  Array[Byte](0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte, 0xBE.toByte)
anObject.someMethod(3, 3.2f, 0xEF.toByte, "3") shouldBe
  Array[Byte](0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte, 0xCD.toByte)

在scala 2.12.4中测试

2 个答案:

答案 0 :(得分:3)

您可以使用隐式转换执行此操作。

在:

def f(b: Byte) = println(s"Byte = $b")
f(0x34)
f(0xFF) // compilation error

后:

implicit def int2Byte(i: Int) = i.toByte

def f(b: Byte) = println(s"Byte = $b")
f(0x34)
f(0xFF)

输出:

Byte = 52
Byte = -1

答案 1 :(得分:3)

回想一下,在Scala中,我们可以使用" pimp-my-library" -pattern,通过向特殊类String添加方法,轻松定义解释任意StringContext文字的新方法。 。例如,我们可以添加方法b来创建单个字节到StringContext,这样我们就可以按如下方式写下字节:

val myByte = b"5C"

以下是如何实施的:

implicit class SingleByteContext(private val sc: StringContext) {
   def b(args: Any*): Byte = {
     val parts = sc.parts.toList
     assert(
       parts.size == 1 && args.size == 0, 
       "Expected a string literal with exactly one part"
     )
     Integer.parseInt(parts(0), 16).toByte
   }
}

为了使用这种新语法,我们必须将上述对象导入隐式范围。然后我们可以这样做:

/* Examples */ {

  def printByte(b: Byte) = println("%02X".format(b))

  printByte(b"01")
  printByte(b"7F")
  printByte(b"FF")
  printByte(b"80")
}

这将打印:

01
7F
FF
80

您可以明显调整实施(例如,您可以重命名" b"到" hex"或" x" 或"牛"或类似的东西)。

请注意,可以轻松扩展此技术以处理整个字节数组,如in this answer to a similar question所述。这将允许您写下字节数组而不重复恼人的0x - 前缀,例如:

val myBytes = hexBytes"AB,CD,5E,7F,5A,8C,80,BC"