欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Rust動態(tài)調(diào)用字符串定義的Rhai函數(shù)方式

 更新時間:2025年02月26日 11:29:45   作者:許野平  
Rust中使用Rhai動態(tài)調(diào)用字符串定義的函數(shù),通過eval_expression_with_scope實現(xiàn),但參數(shù)傳遞和函數(shù)名處理有局限性,使用FnCall功能更健壯,但更復(fù)雜,總結(jié)提供了更通用的方法,但需要處理更多錯誤情況

Rust動態(tài)調(diào)用字符串定義的Rhai函數(shù)

在 Rust 中使用 Rhai 腳本引擎時,你可以動態(tài)地調(diào)用傳入的字符串表示的 Rhai 函數(shù)。

Rhai 是一個嵌入式腳本語言,專為嵌入到 Rust 應(yīng)用中而設(shè)計。

這是一個基本示例

展示了如何在 Rust 中調(diào)用用字符串傳入的 Rhai 函數(shù)。

首先,確保你已經(jīng)將 Rhai 添加到你的 Cargo.toml 文件中:

[dependencies]
rhai = "0.19"  # 請檢查最新版本號

然后,你可以使用以下代碼來調(diào)用用字符串傳入的 Rhai 函數(shù):

use rhai::{Engine, EvalAltResult, FnPtr, Module, Scope};

fn main() -> Result<(), Box<dyn std::error::Error>> {
    // 創(chuàng)建一個 Rhai 引擎實例
    let mut engine = Engine::new();

    // 定義一個 Rhai 模塊,其中包含一些函數(shù)
    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)?;

    // 創(chuàng)建一個作用域
    let mut scope = Scope::new();

    // 示例:要調(diào)用的函數(shù)名及其參數(shù)
    let function_name = "greet".to_string();
    let args: Vec<Box<dyn FnPtr>> = vec![Box::new(|_| "World".to_string()) as Box<dyn FnPtr>];

    // 調(diào)用函數(shù)
    let result: EvalAltResult = engine.eval_expression_with_scope(
        &format!("({})", function_name),
        &mut scope,
        args.iter().cloned().collect::<Vec<_>>(),
    )?;

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

    Ok(())
}

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

  1. 參數(shù)傳遞:在上面的示例中,參數(shù)傳遞是通過創(chuàng)建一個 FnPtr 的向量并傳遞給 eval_expression_with_scope 實現(xiàn)的。但這種方法比較繁瑣,并且只適用于簡單的函數(shù)簽名。
  2. 函數(shù)名處理:函數(shù)名是通過字符串格式化直接嵌入到表達式中的,這意味著你需要確保傳入的函數(shù)名是安全的(即不會導(dǎo)致 Rhai 執(zhí)行不安全的代碼)。

一個更健壯的方法是使用 Rhai 的 FnCall 功能,但這需要更多的設(shè)置和錯誤處理。

這是一個更通用的方法,但稍微復(fù)雜一些

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>> {
    // 創(chuàng)建一個 Rhai 引擎實例
    let mut engine = Engine::new();

    // 定義一個 Rhai 模塊,其中包含一些函數(shù)
    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)?;

    // 創(chuàng)建一個作用域
    let mut scope = Scope::new();

    // 示例:要調(diào)用的函數(shù)名及其參數(shù)
    let call_args = CallArgs {
        func: "greet".to_string(),
        args: vec!["Alice".to_string()],
    };

    // 將參數(shù)轉(zhuǎn)換為 Rhai 值
    let rhai_args: rhai::Array = call_args.args.into_iter().map(|arg| rhai::Value::from(arg)).collect();

    // 定義一個臨時的 Rhai 函數(shù)來調(diào)用目標函數(shù)
    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
    );

    // 調(diào)用函數(shù)
    let result: EvalAltResult = engine.eval_expression(&call_code, &mut scope)?;

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

    Ok(())
}

在這個更通用的示例中,我們定義了一個 CallArgs 結(jié)構(gòu)體來存儲函數(shù)名和參數(shù),然后構(gòu)建了一個臨時的 Rhai 腳本,該腳本根據(jù)函數(shù)名和參數(shù)數(shù)量調(diào)用相應(yīng)的 Rhai 函數(shù)。

這種方法提供了更大的靈活性,但也更復(fù)雜,并且需要處理更多的錯誤情況。

總結(jié)

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 利用Rust實現(xiàn)一個簡單的Ping應(yīng)用

    利用Rust實現(xiàn)一個簡單的Ping應(yīng)用

    這兩年Rust火的一塌糊涂,甚至都燒到了前端,再不學(xué)習(xí)怕是要落伍了。最近翻了翻文檔,寫了個簡單的Ping應(yīng)用練練手,感興趣的小伙伴可以了解一下
    2022-12-12
  • C++的替代:微軟如何使用rust?

    C++的替代:微軟如何使用rust?

    這篇文章主要介紹了微軟如何使用rust的,幫助大家了解c++和rust這兩門編程語言的聯(lián)系與區(qū)別,感興趣的朋友可以了解下
    2020-09-09
  • Rust語言的新手了解和學(xué)習(xí)入門啟蒙教程

    Rust語言的新手了解和學(xué)習(xí)入門啟蒙教程

    這篇文章主要介紹了rust的特點、安裝、項目結(jié)構(gòu)、IDE環(huán)境配置、代碼運行,講解了如何安裝Rust編譯器,創(chuàng)建和運行第一個Rust程序,并對Rust語言的特點和優(yōu)勢作了說明,包括內(nèi)存安全、高效性能、并發(fā)性、社區(qū)支持和統(tǒng)一包管理等,是新手了解和學(xué)習(xí)Rust語言的啟蒙教程
    2024-12-12
  • Rust實現(xiàn)構(gòu)建器模式和如何使用Bon庫中的構(gòu)建器

    Rust實現(xiàn)構(gòu)建器模式和如何使用Bon庫中的構(gòu)建器

    這篇文章主要介紹了Rust實現(xiàn)構(gòu)建器模式和如何使用Bon庫中的構(gòu)建器,本文給大家介紹的非常詳細,感興趣的朋友跟隨小編一起看看吧
    2024-08-08
  • Rust開發(fā)WebAssembly在Html和Vue中的應(yīng)用小結(jié)(推薦)

    Rust開發(fā)WebAssembly在Html和Vue中的應(yīng)用小結(jié)(推薦)

    這篇文章主要介紹了Rust開發(fā)WebAssembly在Html和Vue中的應(yīng)用,本文將帶領(lǐng)大家在普通html上和vue手腳架上都來運行wasm的流程,需要的朋友可以參考下
    2022-08-08
  • Rust在寫庫時實現(xiàn)緩存的操作方法

    Rust在寫庫時實現(xiàn)緩存的操作方法

    Moka是一個用于Rust的高性能緩存庫,它提供了多種類型的緩存數(shù)據(jù)結(jié)構(gòu),包括哈希表、LRU(最近最少使用)緩存和?支持TTL(生存時間)緩存,這篇文章給大家介紹Rust在寫庫時實現(xiàn)緩存的相關(guān)知識,感興趣的朋友一起看看吧
    2024-01-01
  • 深入了解Rust的生命周期

    深入了解Rust的生命周期

    生命周期指的是引用保持有效的作用域,Rust?的每個引用都有自己的生命周期。本文將通過示例和大家詳細說說Rust的生命周期,需要的可以參考一下
    2022-11-11
  • 在Rust?web服務(wù)中使用Redis的方法

    在Rust?web服務(wù)中使用Redis的方法

    這篇文章主要介紹了在Rust?web服務(wù)中使用Redis,在這篇文章中,我們將演示如何在一個Rust?web應(yīng)用程序中使用Redis,需要的朋友可以參考下
    2022-08-08
  • Rust中類型轉(zhuǎn)換在錯誤處理中的應(yīng)用小結(jié)

    Rust中類型轉(zhuǎn)換在錯誤處理中的應(yīng)用小結(jié)

    隨著項目的進展,關(guān)于Rust的故事又翻開了新的一頁,今天來到了服務(wù)器端的開發(fā)場景,發(fā)現(xiàn)錯誤處理中的錯誤類型轉(zhuǎn)換有必要分享一下,對Rust錯誤處理相關(guān)知識感興趣的朋友一起看看吧
    2023-09-09
  • rust的nutyp驗證和validator驗證數(shù)據(jù)的方法示例詳解

    rust的nutyp驗證和validator驗證數(shù)據(jù)的方法示例詳解

    本文介紹了在Rust語言中,如何使用nuType和validator兩種工具來對Cargo.toml和modules.rs文件進行驗證,通過具體的代碼示例和操作步驟,詳細解釋了驗證過程和相關(guān)配置,幫助讀者更好地理解和掌握使用這兩種驗證工具的方法,更多Rust相關(guān)技術(shù)資訊,可繼續(xù)關(guān)注腳本之家
    2024-09-09

最新評論