当使用 V8 作为脚本引擎时,我向 Javascript 公开了一个名为 construct_with_ec6_syntax 的 C++ 函数。这个函数在被调用时应该简单地返回一个 some_ec6_class 的实例。
这个 C++ 函数基本上应该执行以下等效的 Javascript:
return new some_ec6_class(111, 222);
这个类将在 Javascript 中定义如下 EC6 语法:
class some_ec6_class
{
constructor(x, y) {
this.result = x + y;
}
}
我的目标是在 Javascript 中运行以下...
var the_instance = construct_with_ec6_syntax(111, 222);
the_instance.result ; // .. and get 333 here.
我当前的 C++ 函数实现是这样的:
void construct_with_ec6_syntax(const FunctionCallbackInfo<Value>& args) {
Handle<Object> global = args.GetIsolate()->GetCurrentContext()->Global();
std::string newvecfunc = "some_ec6_class";
Handle<Value> value = global->Get(String::NewFromUtf8(args.GetIsolate(), newvecfunc.c_str(), String::kNormalString, newvecfunc.length()));
Local<Value> result;
if (value->IsFunction()) {
Handle<Function> func = Handle<Function>::Cast(value);
Handle<Value> args2[2];
args2[0] = Number::New(args.GetIsolate(), 111);
args2[1] = Number::New(args.GetIsolate(), 222);
result = func->CallAsConstructor(args.GetIsolate()->GetCurrentContext(), 2, args2).ToLocalChecked();
}
args.GetReturnValue().Set(result);
}
从 Javascript 运行这个函数会返回 undefined!而不是我期望的对象。正如 xaxxon 向我指出的那样,这是因为 value->IsFunction() 返回 false,但 value->IsUndefined() 返回 true。如果我使用 with non EC6 语法定义了类,则上述函数确实返回一个实例..
function some_non_ec6_class(x, y) // this guy would work with the above function
{
this.result = x + y;
}
所以我有点困惑!我是否需要更具体一些,比如首先从对象中获取 constructor 函数,然后调用 CallAsConstructor?
感谢任何提示!
(这个问题看起来类似于 Calling a v8 javascript function from c++ with an argument但不同。)
我从 2016 年 10 月 22 日开始使用 V8 结账。 完整的测试用例:
https://gist.github.com/rayburgemeestre/c0abd528f6f67edbfe686d484c45ddbb
次要更新:
正如您在评论中看到的,我还做了一个更具体的关于从上下文中“获取”类的测试用例:https://gist.github.com/rayburgemeestre/df6193d532c7b7908fe27c89799bfa3a
我还发布到 v8-users 邮件列表:https://groups.google.com/forum/#!topic/v8-users/Hj2j4rJMwBw
最佳答案
class 是javascript中创建变量的一种方式,类似于let。使用 class 创建的变量是 block 范围的,不会创建全局属性(同样,类似于 let)。因此它在 context.global 中不可用,就像一个函数是:
http://exploringjs.com/es6/ch_variables.html
function Complete Block Yes
class No Block No
您需要为您的代码明确创建一个全局变量:
some_es6_class = class{};
或者在使用“传统 es6”语法创建类对象后显式地将类对象传递给您的 C++ 函数:
class some_es6_class{};
some_native_function(some_es6_class);
编辑:我做了更多的挖掘,我相信上下文对象有一个 LexicalEnvironment,它在它的 script 中共享,但与全局对象不同.查找名称时,它会遍历 LexicalEnvironments 的父层次结构以查找该名称。这些词法环境不会通过 API 公开(并且可能实际上不存在 - 它们是 JS 规范用来定义行为的构造,而不是实现的必需部分)
关于javascript - 如何从 C++ 中返回一个新的 V8 javascript "class"实例?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41211533/
我正在学习如何使用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但我想要一些方法来使用
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
给定这段代码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
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
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
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta