1

20.3.6 多态变体”小节描述了如何在 C 中识别多态变体值(*它包含一个错误:应该是caml_hash_variant而不是hash_variant

我想直接将这些哈希值用作 C++ 中的错误代码。类似的东西

存档.mli:

...
type t = ...
type err = [`File_not_found | `Archive_is_corrupted]
val opena : string -> (t, err) error
... 

存档.ml

...
let () = Callback.register "open archive" opena
...

存档.cpp:

...
const int Error::File_not_found = caml_hash_variant("File_not_found")
const int Error::Archive_is_corrupted = caml_hash_variant("Archive_is_corrupted")

int Archive::open(char* path) {
  static const value* f = nullptr; \
  if (f = nullptr) 
      f = caml_named_value("open archive");

  value result = caml_callback(*f, caml_copy_string(path));
  if (Tag_val(result) == 0) { // Result.Ok
    archive = Field(caml_state, 0);
    return ??????
  } else { // Result.Error
    return Int_val(Field(caml_state, 0));
  }
}
...

返回错误代码并进行比较没有问题

if (x.open(path) == Error::Archive_is_corrupted) {
...
}

但我不知道我可以返回什么状态0? -1?
是否存在无法返回的保证值caml_hash_variant

4

1 回答 1

0

通常 OCaml 实现中的立即数设置了低位,而变体哈希是立即数。因此,如果您正在查看 C++ 中的变体哈希值,您可以确定 caml_hash_variant 永远不会返回值 0。

如果查看代码,最终值是由 Val_int() 或 Val_long() 生成的。在这些宏的定义中,您会看到它们保证设置了低位。

我没有对代码进行任何类型的分析,但值 -1 至少在表面上可能作为哈希值,因为它的低位已设置。

更新

立即值的低位设置为垃圾收集器的标记。所以这是一个必须严格遵守的约定。(恕我直言,这是 OCaml 实现中许多非常好的设计权衡之一。)

于 2020-07-22T23:34:54.337 回答