问题描述
我正在尝试掌握 ASN.1 的概念和编码方案(比如 DER),并希望进行完整性检查。
到目前为止我得到的是ASN.1定义了抽象数据类型(INTEGER、BOOLEAN等,有些更复杂),它允许你以抽象的方式定义数据结构,例如:
World-Schema DEFinitioNS AUTOMATIC TAGS ::=
BEGIN
Rocket ::= SEQUENCE
{
range INTEGER,-- huge (see a special directive above)
name UTF8String (SIZE(1..16)),message UTF8String DEFAULT "Hello World",fuel ENUMERATED {solid,liquid,gas},speed CHOICE
{
mph INTEGER,kmph INTEGER
} OPTIONAL,payload SEQUENCE OF UTF8String
}
END
这通过为其定义某些字段来定义“火箭”的结构。 DER 作为一种编码方案,只有当我想编码一个对象时才起作用,即这个数据结构的具体值。所以基本上我可以以某种方式告诉我的电脑:根据这个 ASN.1 模块,序列化以下 JSON 文件:
{
"range":340282366920938463463374607431768211455,"name":"Falcon","fuel":"solid","speed":{"mph":18000},"payload":["Car","GPS"]
}
它会生成符合特定标准的比特流,该标准由 DER 标准定义。 (为了方便起见,X.690)。现在这个比特流的每个接收者(知道它是一个 DER 编码的数据)将知道根据 ASN.1 解释这个消息。
我无法真正向自己解释的一件事是:我们必须定义一个接口吗?我的意思是,一旦我们有了定义数据类型(INTEGER、STRING 等)的语言 (ASN.1) - 就可以组合消息、编码它们并发送它们,并且它们可以被理解。为什么定义我们之间的接口很重要?
如果我理解错了,请纠正我。
解决方法
ASN.1 不是一种语言,首字母缩写词代表 Abstract Syntax Notation。
ASN.1 是一套规范,旨在消除两个想要相互通信的系统之间的所有歧义。
首先,您描述将要传输的内容(传输语法)。在您的示例中:具有射程、名称等的火箭......
然后,您决定将如何发送内容(编码规则)。
因为您显示了一个 Json 值(它是人类可读的),所以您可以认为该值可以解释自身。
但是如果你采用二进制编码规则(这是 ASN.1 的本质),你很快就会意识到你绝对需要知道传输语法(你称之为接口的东西)才能解码一个值。
某些二进制编码规则(如 BER、DER、CER)给人的印象是您可以在不知道规范的情况下使用它们,因为它们遵循标签/长度/值模式……如果您知道如何解码标签并且一个长度,你可以做一个最小的
但是,当您需要使用打包编码规则 (PER) 时,如果没有规范……和好的工具,您将完全无能为力。