我正在调用@JSFunction
ScriptableObject
注释方法
JavaScript文件
Target = Packages.com.acme.rhino.Target;
function evaluate() {
var t = Target();
t.addModifier("foobar", 1);
return t;
}
Java文件
public class Target extends ScriptableObject {
private static final long serialVersionUID = 1L;
public List<Modifier> modifiers = new LinkedList<>();
@JSConstructor
public Target() {
}
@JSFunction
public void addModifier(final String message, final int value) {
modifiers.add(new Modifier(message, value));
}
public int getValue() {
int sum = 0;
for (final Modifier modifier : modifiers) {
sum += modifier.getValue();
}
return sum;
}
@Override
public String getClassName() {
return "Target";
}
}
但是我得到了
org.mozilla.javascript.EcmaError: TypeError: Cannot find default value for object.
at org.mozilla.javascript.ScriptRuntime.constructError(ScriptRuntime.java:3687)
at org.mozilla.javascript.ScriptRuntime.constructError(ScriptRuntime.java:3665)
at org.mozilla.javascript.ScriptRuntime.typeError(ScriptRuntime.java:3693)
at org.mozilla.javascript.ScriptRuntime.typeError1(ScriptRuntime.java:3705)
at org.mozilla.javascript.ScriptableObject.getDefaultValue(ScriptableObject.java:976 )
at org.mozilla.javascript.ScriptableObject.getDefaultValue(ScriptableObject.java:895 )
at org.mozilla.javascript.ScriptRuntime.toString(ScriptRuntime.java:761)
at org.mozilla.javascript.ScriptRuntime.notFunctionError(ScriptRuntime.java:3774)
at org.mozilla.javascript.ScriptRuntime.getPropFunctionAndThisHelper(ScriptRuntime. java:2269)
at org.mozilla.javascript.ScriptRuntime.getPropFunctionAndThis(ScriptRuntime. java:2251)
at org.mozilla.javascript.optimizer.OptRuntime.callProp0(OptRuntime.java:83)
at org.mozilla.javascript.gen.script_5._c_evaluate_1(script:6)
at org.mozilla.javascript.gen.script_5.call(script)
at org.mozilla.javascript.ContextFactory.doTopCall(ContextFactory.java:394)
at org.mozilla.javascript.ScriptRuntime.doTopCall(ScriptRuntime.java:3091)
at org.mozilla.javascript.gen.script_5.call(script)
并且不知道从哪里去。当我不调用addModifier
方法时,给定的代码有效,并且在堆栈跟踪中给出错误notFunctionError
,我认为Rhino不会将给定的方法解释为JavaScript函数。
可以找到重现错误的完整Maven项目here
答案 0 :(得分:3)
上述方法的问题是在脚本范围内未正确设置Target.prototype
。有关如何在脚本范围内正确定义原型的详细信息,请参阅静态ScriptableObject.defineClass()
方法。
您可以为脚本提供Target
构造函数。 first alternative将始终为所有脚本定义Target
构造函数。如果您事先知道希望Target
全局可用,则此方法很有效。这基本上归结为以下几点:
final Context context = Context.enter();
try {
final ScriptableObject scope = context.initStandardObjects();
ScriptableObject.defineClass(scope, Target.class, false, true);
context.evaluateString(scope, script, "script", 1, null);
// etc.
} finally {
Context.exit();
}
如果您希望脚本作者决定哪些构造函数是必需的,second alternative是为脚本提供defineClass
函数。使用此函数,脚本作者可以在其类路径上“导入”任何可编写脚本的对象(这可能超出您想要允许的范围)。要向脚本提供defineClass
函数,请在输入上下文后执行以下操作:
final Context context = Context.enter();
try {
final ScriptableObject scope = context.initStandardObjects();
scope.defineFunctionProperties(
new String[] {"defineClass"},
Global.class,
ScriptableObject.DONTENUM);
context.evaluateString(scope, script, "script", 1, null);
// etc.
} finally {
Context.exit();
}
然后,JavaScript作者使用Target
构造函数,其中包含以下内容:
defineClass("com.acme.rhino.Target");
// whatever `getClassName()` returns is now available
var target = new Target();
在上述两个分支中,如果您向Target
构造函数添加更多内容,我还做了一些其他更改,这些更改可以让您更好。零参数构造函数不需要@JSConstructor
注释。如果您以后想要一个接受参数的构造函数,那么这个零参数构造函数将被用作原型构造函数,并且您可以在将用于初始化对象的方法上使用@JSConstructor
注释。根据您编写此构造函数方法的方式,在JavaScript中使用new
关键字非常重要。
简而言之,Packages.com.acme...
语法 对于从脚本访问ScriptableObject
构造函数非常有用。
答案 1 :(得分:0)
我通过使用new运算符来实现它(非常相似的代码)。在你的例子中做
function evaluate() {
var t = new Target();
...
应该也可以。