在CPP中调用Jscript中的函数

本文介绍如何使用C++通过msscript.ocx控件调用JScript函数。通过创建IScriptControl对象实例并设置语言为JScript,可以执行JScript代码或调用特定函数。文章提供了完整的示例代码。

CPP中调用Jscript中的函数

C++中调用Jscript的函数非常简单,Windows提供了一个msscript.ocx的控件,利用这个控件可以直接操作Jscript: 执行一段Jscript脚本,或者调用指定的函数。我写了一个简单的例子:

l 导入msscript.ocx。下面这条指令会在项目目录中生成msscript.tlimsscript.tlh两个文件,里面有msscript.ocx中所有接口的描述和IID的定义。

#import "msscript.ocx" no_namespace

l 声明一个对象。

CComPtr<IScriptControl> m_iScriptControl;

l 创建对象实例

if(SUCCEEDED(m_iScriptControl.CoCreateInstance(__uuidof(ScriptControl))))

l 设置语言等属性。

m_iScriptControl->PutLanguage(L"JScript");

m_iScriptControl->PutAllowUI(VARIANT_FALSE);

l 加入Jscript代码。

m_iScriptControl->AddCode(L"function test(str1, str2) { return str1 + /"-ok-/" + str2; }");

l 获得函数,这里要说明的是GetItem的参数是1n,而不是0n-1

CComPtr<IScriptProcedureCollection> aProcedureSet = m_iScriptControl->GetProcedures();

long n = aProcedureSet->GetCount();

CComPtr<IScriptProcedure> aProcedure = aProcedureSet->GetItem(_variant_t(n));

_bstr_t strFunction = aProcedure->GetName();

l 准备函数参数。

VARIANT va = {0};

va.vt = VT_BSTR;

n = 2;

SAFEARRAYBOUND bounds[1] = {0};

bounds[0].cElements = n;

SAFEARRAY* psa = SafeArrayCreate(VT_VARIANT, 1, bounds);

long i = 0;

for(i = 0; i < n; i++)

{

va.bstrVal = SysAllocString(L"test");

SafeArrayPutElement(psa, &i, &va);

}

l 调用函数。

_variant_t Result = m_iScriptControl->Run(strFunction, &psa);

l 释放参数。

for(i = 0; i < n; i++)

{

va.bstrVal = SysAllocString(L"test");

SafeArrayGetElement(psa, &i, &va);

SysFreeString(va.bstrVal);

}

SafeArrayDestroy(psa);

在 C++ 中,`acedEvaluate` 是 AutoCAD ObjectARX 开发环境中的一个函数,用于执行 LISP 表达式并返回结果。若需寻找具有类似功能的函数或方法,可以从以下几个方向进行探索: ### 1. 使用 `acutEvalExpr` 执行 LISP 表达式 `acutEvalExpr` 是另一个与 `acedEvaluate` 功能相近的函数,它同样用于执行 LISP 表达式,但通常用于非交互式的上下文环境中。其原型如下: ```cpp int acutEvalExpr(const char* lispCode, resbuf** pResult); ``` 与 `acedEvaluate` 不同的是,`acutEvalExpr` 更适用于在后台处理表达式,不依赖于当前编辑器状态。例如: ```cpp resbuf* pResult = nullptr; if (acutEvalExpr("(+ 1 2)", &pResult) == RTNORM) { acutPrintf(L"\nResult: %d", pResult->resval.rint); acutRelRb(pResult); } ``` 此方式适用于需要在 C++ 中嵌入 LISP 脚本逻辑的场景[^5]。 --- ### 2. 使用 `acedInvoke` 调用 AutoLISP 函数 若目标是调用已定义的 AutoLISP 函数而非直接执行字符串表达式,可以使用 `acedInvoke` 函数。该函数允许传递参数并接收返回值,常用于与用户自定义的 LISP 函数交互。示例代码如下: ```cpp resbuf* args = acutBuildList(RTSTR, "myFunctionName", RTNONE); resbuf* result = nullptr; acedInvoke(args, &result); acutRelRb(args); ``` 这种方式更贴近传统函数调用模式,适用于已有 LISP 函数封装的场景[^5]。 --- ### 3. 使用脚本引擎接口(如 JavaScript 或 .NET) 对于现代 AutoCAD 插件开发,也可以通过集成外部脚本引擎(如 JScript、VBScript 或 .NET)来实现类似功能。例如,在 .NET 环境中,可以通过 `CommandStack` 和 `ExecuteCommand` 方法模拟命令行输入: ```csharp Document doc = Application.DocumentManager.MdiActiveDocument; Database db = doc.Database; Editor ed = doc.Editor; PromptKeywordOptions pko = new PromptKeywordOptions("Select option:", "Yes No"); PromptKeywordResult pkr = ed.GetKeywords(pko); ``` 虽然不属于原生 C++ 的范畴,但在某些扩展性项目中可能提供更灵活的替代方案。 --- ### 4. 使用 `system()` 或 `exec()` 系列函数(不推荐) 在标准 C++ 中,也可以通过调用系统命令的方式间接执行脚本语言(如 Python、Lua 等),从而模拟 `acedEvaluate` 的效果。例如: ```cpp #include <cstdlib> int result = system("python -c \"print(1 + 2)\""); ``` 然而,此类方法存在安全风险且不适用于嵌入式插件开发,因此不建议用于 AutoCAD 插件开发流程中。 --- ### 总结 | 函数/方法 | 适用场景 | 特点说明 | |--------------------|--------------------------------------|------------------------------------------------| | `acedEvaluate` | 执行 LISP 字符串表达式 | 交互性强,依赖当前编辑器上下文 | | `acutEvalExpr` | 后台执行 LISP 表达式 | 非交互式,适合自动化任务 | | `acedInvoke` | 调用已注册的 LISP 函数 | 支持参数传递和返回值 | | 外部脚本引擎 | 集成其他语言逻辑 | 灵活但需额外配置 | | `system()` / `exec()` | 调用外部脚本解释器 | 安全性低,不推荐用于插件开发 | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值