Rust:如何动态调用字符串定义的 Rhai 函数?

在 Rust 中使用 Rhai 脚本引擎时,你可以动态地调用传入的字符串表示的 Rhai 函数。Rhai 是一个嵌入式脚本语言,专为嵌入到 Rust 应用中而设计。以下是一个基本示例,展示了如何在 Rust 中调用用字符串传入的 Rhai 函数。

首先,确保你已经将 Rhai 添加到你的 Cargo.toml 文件中:

toml 复制代码
[dependencies]
rhai = "0.19"  # 请检查最新版本号

然后,你可以使用以下代码来调用用字符串传入的 Rhai 函数:

rust 复制代码
use rhai::{Engine, EvalAltResult, FnPtr, Module, Scope};

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // 创建一个 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<Box<dyn FnPtr>> = vec![Box::new(|_| "World".to_string()) as Box<dyn FnPtr>];

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

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

    Ok(())
}

然而,上面的代码有一些限制和简化的地方:

  1. 参数传递 :在上面的示例中,参数传递是通过创建一个 FnPtr 的向量并传递给 eval_expression_with_scope 实现的。但这种方法比较繁琐,并且只适用于简单的函数签名。
  2. 函数名处理:函数名是通过字符串格式化直接嵌入到表达式中的,这意味着你需要确保传入的函数名是安全的(即不会导致 Rhai 执行不安全的代码)。

一个更健壮的方法是使用 Rhai 的 FnCall 功能,但这需要更多的设置和错误处理。以下是一个更通用的方法,但稍微复杂一些:

rust 复制代码
use rhai::{Engine, EvalAltResult, Module, Scope};
use rhai::serde::{Deserialize, Serialize};

#[derive(Debug, Serialize, Deserialize)]
struct CallArgs {
    func: String,
    args: Vec<String>,
}

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // 创建一个 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) -> Any {{
            let func = match func_name.as_str() {{
                "greet" => greet,
                "add" => add as fn(i32, i32) -> i32,
                _ => return "Function not found".into(),
            }};
            
            match (func, args.len()) {{
                (greet, 1) => greet(args[0].cast::<String>()?),
                (add, 2) => add(args[0].cast::<i32>()?, args[1].cast::<i32>()?),
                _ => return "Invalid argument count".into(),
            }}
        }}
        call_func("{}", {})
        "#,
        call_args.func, rhai_args
    );

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

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

    Ok(())
}

在这个更通用的示例中,我们定义了一个 CallArgs 结构体来存储函数名和参数,然后构建了一个临时的 Rhai 脚本,该脚本根据函数名和参数数量调用相应的 Rhai 函数。这种方法提供了更大的灵活性,但也更复杂,并且需要处理更多的错误情况。

相关推荐
七灵微17 分钟前
【后端】Flask
后端·python·flask
SomeB1oody19 分钟前
【Rust自学】17.2. 使用trait对象来存储不同值的类型
开发语言·后端·rust
gopher_looklook1 小时前
从零到一: 用Go语言搭建简易RPC框架并实践 (一)
后端·go
SomeB1oody1 小时前
【Rust自学】17.3. 实现面向对象的设计模式
开发语言·设计模式·rust
慕璃嫣2 小时前
Haskell语言的安全开发
开发语言·后端·golang
qq_544329173 小时前
CRM项目的开发与调试整体策略
前端·后端·bug
黄同学real7 小时前
使用.NET 8构建高效的时间日期帮助类
后端·c#·.net
ChinaRainbowSea8 小时前
四.4 Redis 五大数据类型/结构的详细说明/详细使用( zset 有序集合数据类型详解和使用)
java·javascript·数据库·redis·后端·nosql
eybk11 小时前
Qpython+Flask监控添加发送语音中文信息功能
后端·python·flask