虚荣 發表於 2025-2-26 11:29:45

Rust动态调用字符串定义的Rhai函数方式

<div id="navCategory"><h5 class="catalogue">目录</h5><ul class="first_class_ul"><li>Rust动态调用字符串定义的Rhai函数</li><ul class="second_class_ul"><li>这是一个基本示例</li><li>这是一个更通用的方法,但稍微复杂一些</li></ul><li>总结</li><ul class="second_class_ul"></ul></ul></div><p class="maodian"></p><h2>Rust动态调用字符串定义的Rhai函数</h2>
<p>在 Rust 中使用 Rhai 脚本引擎时,你可以动态地调用传入的字符串表示的 Rhai 函数。</p>
<p>Rhai 是一个嵌入式脚本语言,专为嵌入到 Rust 应用中而设计。</p>
<p class="maodian"></p><h3>这是一个基本示例</h3>
<p>展示了如何在 Rust 中调用用字符串传入的 Rhai 函数。</p>
<p>首先,确保你已经将 Rhai 添加到你的 <code>Cargo.toml</code> 文件中:</p>
<div class="jb51code"><pre class="brush:bash;">
rhai = "0.19"# 请检查最新版本号</pre></div>
<p>然后,你可以使用以下代码来调用用字符串传入的 Rhai 函数:</p>
<div class="jb51code"><pre class="brush:bash;">use rhai::{Engine, EvalAltResult, FnPtr, Module, Scope};

fn main() -&gt; Result&lt;(), Box&lt;dyn std::error::Error&gt;&gt; {
    // 创建一个 Rhai 引擎实例
    let mut engine = Engine::new();

    // 定义一个 Rhai 模块,其中包含一些函数
    let mut module = Module::new();
    module.insert_fn("greet", |name: String| format!("Hello, {}", name));
    module.insert_fn("add", |a: i32, b: i32| a + b);

    // 将模块注册到引擎中
    engine.register_module(module)?;

    // 创建一个作用域
    let mut scope = Scope::new();

    // 示例:要调用的函数名及其参数
    let function_name = "greet".to_string();
    let args: Vec&lt;Box&lt;dyn FnPtr&gt;&gt; = vec!;

    // 调用函数
    let result: EvalAltResult = engine.eval_expression_with_scope(
      &amp;format!("({})", function_name),
      &amp;mut scope,
      args.iter().cloned().collect::&lt;Vec&lt;_&gt;&gt;(),
    )?;

    // 打印结果
    match result {
      EvalAltResult::Value(value) =&gt; println!("Result: {}", value.render()?),
      _ =&gt; println!("Result is not a value"),
    }

    Ok(())
}</pre></div>
<p>然而,上面的代码有一些限制和简化的地方:</p>
<ol><li><strong>参数传递</strong>:在上面的示例中,参数传递是通过创建一个 <code>FnPtr</code> 的向量并传递给 <code>eval_expression_with_scope</code> 实现的。但这种方法比较繁琐,并且只适用于简单的函数签名。</li><li><strong>函数名处理</strong>:函数名是通过字符串格式化直接嵌入到表达式中的,这意味着你需要确保传入的函数名是安全的(即不会导致 Rhai 执行不安全的代码)。</li></ol>
<p>一个更健壮的方法是使用 Rhai 的 <code>FnCall</code> 功能,但这需要更多的设置和错误处理。</p>
<p class="maodian"></p><h3>这是一个更通用的方法,但稍微复杂一些</h3>
<div class="jb51code"><pre class="brush:bash;">use rhai::{Engine, EvalAltResult, Module, Scope};
use rhai::serde::{Deserialize, Serialize};

#
struct CallArgs {
    func: String,
    args: Vec&lt;String&gt;,
}

fn main() -&gt; Result&lt;(), Box&lt;dyn std::error::Error&gt;&gt; {
    // 创建一个 Rhai 引擎实例
    let mut engine = Engine::new();

    // 定义一个 Rhai 模块,其中包含一些函数
    let mut module = Module::new();
    module.insert_fn("greet", |name: String| format!("Hello, {}", name));
    module.insert_fn("add", |a: i32, b: i32| a + b);

    // 将模块注册到引擎中
    engine.register_module(module)?;

    // 创建一个作用域
    let mut scope = Scope::new();

    // 示例:要调用的函数名及其参数
    let call_args = CallArgs {
      func: "greet".to_string(),
      args: vec!["Alice".to_string()],
    };

    // 将参数转换为 Rhai 值
    let rhai_args: rhai::Array = call_args.args.into_iter().map(|arg| rhai::Value::from(arg)).collect();

    // 定义一个临时的 Rhai 函数来调用目标函数
    let call_code = format!(
      r#"
      fn call_func(func_name: String, args: Array) -&gt; Any {{
            let func = match func_name.as_str() {{
                "greet" =&gt; greet,
                "add" =&gt; add as fn(i32, i32) -&gt; i32,
                _ =&gt; return "Function not found".into(),
            }};
            
            match (func, args.len()) {{
                (greet, 1) =&gt; greet(args.cast::&lt;String&gt;()?),
                (add, 2) =&gt; add(args.cast::&lt;i32&gt;()?, args.cast::&lt;i32&gt;()?),
                _ =&gt; return "Invalid argument count".into(),
            }}
      }}
      call_func("{}", {})
      "#,
      call_args.func, rhai_args
    );

    // 调用函数
    let result: EvalAltResult = engine.eval_expression(&amp;call_code, &amp;mut scope)?;

    // 打印结果
    match result {
      EvalAltResult::Value(value) =&gt; println!("Result: {}", value.render()?),
      _ =&gt; println!("Result is not a value"),
    }

    Ok(())
}</pre></div>
<p>在这个更通用的示例中,我们定义了一个 <code>CallArgs</code> 结构体来存储函数名和参数,然后构建了一个临时的 Rhai 脚本,该脚本根据函数名和参数数量调用相应的 Rhai 函数。</p>
<p>这种方法提供了更大的灵活性,但也更复杂,并且需要处理更多的错误情况。</p>
<p class="maodian"></p><h2>总结</h2>
<p>以上为个人经验,希望能给大家一个参考,也希望大家多多支持琼殿技术社区。</p>
                           
                            <div class="art_xg">
                              <b>您可能感兴趣的文章:</b><ul><li>Rust中的方法与关联函数使用解读</li><li>Rust中的模块系统之控制作用域与私有性详解</li><li>Rust之Rhai脚本编程的示例</li><li>Rust中的&和ref使用解读</li><li>Rust中的注释使用解读</li></ul>
                            </div>

                        </div>
                        <!--endmain-->
頁: [1]
查看完整版本: Rust动态调用字符串定义的Rhai函数方式