我有一个用 C++ 编写的遗留数据结构和一个 OCaml 中的新工具,该工具有望处理该遗留数据。所以我需要将数据从前者导入/翻译到后者。数据以树的形式存在,通常由访问者处理。
作为一个简单的例子,考虑这个最小的 DSL:
#include <memory>
using namespace std;
class intnode;
class addnode;
struct visitor {
virtual void visit(const intnode& n) = 0;
virtual void visit(const addnode& n) = 0;
};
struct node {
virtual void accept(visitor& v) = 0;
};
struct intnode : public node {
int x;
virtual void accept(visitor& v) { v.visit(*this); }
};
struct addnode : public node {
shared_ptr<node> l;
shared_ptr<node> r;
virtual void accept(visitor& v) { v.visit(*this); }
};
它在 OCaml 中的表示如下:
type node = Int of int
| Plus of node * node
let make_int x = Int x
let make_plus l r = Plus(l,r)
问题是,我如何安全有效地将 C++ 树转换为其 OCaml 表示形式?
到目前为止,我有两种方法:
编写一个调用 OCaml 构造函数并产生一个 value 的访问者,例如像这样:
value translate(shared_ptr<node> n);
struct translator : public visitor {
value retval;
virtual visit(const intnode& n) {
retval = call(make_int, Val_int(x->value));
}
virtual visit(const addnode& n) {
value l = translate(n.l);
value r = translate(n.r);
retval = call(make_add, l, r);
}
};
value translate(shared_ptr<node> n)
{
translator t;
t.visit(*n);
}
简单地假设 call 完成所有必需的脚手架以回调 OCaml 并调用正确的构造函数。
该方法的问题在于 OCaml 的垃圾收集器。如果 GC 运行,而 C++ 端在堆栈上有一些 value,该值(毕竟是指向 OCaml 堆的指针)可能会失效。所以我需要一些方法来通知 OCaml 仍然需要这些值这一事实。通常这是通过 CAML* 宏完成的,但在这种情况下我该如何做呢?我可以在 visit 方法中使用这些宏吗?
第二种方法比较复杂。当没有办法安全地存储中间引用时,我可以扭转局面并将 C++ 指针插入 OCaml 堆:
type cppnode (* C++ pointer *)
type functions = {
transl_plus : cppnode -> cppnode -> node;
transl_int : int -> node;
}
external dispatch : functions -> cppnode -> node = "dispatch_transl"
let rec translate n = dispatch {transl_plus; transl_int = make_int} n
and transl_plus a b = make_plus (translate a) (translate b)
这里的想法是,“dispatch”函数将所有子节点包装到 CustomVal 结构中,并将它们传递给 OCaml,而不存储任何中间值。对应的访问者只会实现模式匹配。这显然应该适用于 GC,但缺点是效率稍低(因为指针环绕)和可读性可能较低(因为分派(dispatch)和重建之间的区别)。
有没有办法在方法 1 的优雅的同时获得方法 2 的安全性?
最佳答案
即使在递归情况下,我也没有发现在 C 堆栈上构造 OCaml 值有任何问题。在您的示例中,您使用结构成员来存储 OCaml 堆值。这也是可能的,但是,您需要使用 caml_register_global_root 或 caml_register_generational_root 并使用 caml_remove_global_root 或 caml_remove_generational_global_root 释放它们。事实上,您甚至可以构建一个智能指针来保存 OCaml 值。
尽管如此,我仍然看不出有任何理由(至少对于您演示的简化示例而言)为什么您应该为此进入类(class)成员,这就是我要解决的问题:
struct translator : public visitor {
virtual value visit(const intnode& n) {
CAMLparam0();
CAMLlocal1(x);
x = call(make_int, Val_int(n->value);
CAMLreturn(x);
}
virtual value visit(const addnode& n) {
CAMLparam0();
CAMLlocal(l,r,x);
l = visit(*n.l);
r = visit(*n.r);
x = call(make_add, l, r);
CAMLreturn(x);
}
};
当然,这假设您有一个可以返回任意类型值的访问者。如果您没有,也不想实现,那么您绝对可以逐步建立您的值(value):
value translate(shared_ptr<node> n);
class builder : public visitor {
value result;
public:
builder() {
result = Val_unit; // or any better default
caml_register_generational_global_root(&result);
}
virtual ~builder() {
caml_remove_generational_global_root(&result);
}
virtual void visit(const intnode& n) {
CAMLparam0();
CAMLlocal1(x);
x = call(make_int, Val_int(n->value);
caml_modify_generational_global_root(&result, x);
CAMLreturn0;
}
virtual void visit(const addnode& n) {
CAMLparam0();
CAMLlocal(l,r,x);
l = translate(n.l);
r = translate(n.r);
x = call(make_add, l, r);
caml_modify_generational_global_root(&result,x)
CAMLreturn0;
}
};
value translate(share_ptr<node> node) {
CAMLparam0();
CAMLlocal1(x);
builder b;
b.visit(*node);
x = b.result;
CAMLreturn(x);
}
您还可以查看 Berke Durak 的 Aurochs项目,使用 C 就地构建解析树。
关于c++ - 如何安全地在 C++/Ocaml 之间转换树数据结构?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46562400/
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack