Rust動(dòng)態(tài)調(diào)用字符串定義的Rhai函數(shù)方式
Rust動(dòng)態(tài)調(diào)用字符串定義的Rhai函數(shù)
在 Rust 中使用 Rhai 腳本引擎時(shí),你可以動(dòng)態(tài)地調(diào)用傳入的字符串表示的 Rhai 函數(shù)。
Rhai 是一個(gè)嵌入式腳本語(yǔ)言,專為嵌入到 Rust 應(yīng)用中而設(shè)計(jì)。
這是一個(gè)基本示例
展示了如何在 Rust 中調(diào)用用字符串傳入的 Rhai 函數(shù)。
首先,確保你已經(jīng)將 Rhai 添加到你的 Cargo.toml
文件中:
[dependencies] rhai = "0.19" # 請(qǐng)檢查最新版本號(hào)
然后,你可以使用以下代碼來(lái)調(diào)用用字符串傳入的 Rhai 函數(shù):
use rhai::{Engine, EvalAltResult, FnPtr, Module, Scope}; fn main() -> Result<(), Box<dyn std::error::Error>> { // 創(chuàng)建一個(gè) Rhai 引擎實(shí)例 let mut engine = Engine::new(); // 定義一個(gè) 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); // 將模塊注冊(cè)到引擎中 engine.register_module(module)?; // 創(chuàng)建一個(gè)作用域 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(()) }
然而,上面的代碼有一些限制和簡(jiǎn)化的地方:
- 參數(shù)傳遞:在上面的示例中,參數(shù)傳遞是通過(guò)創(chuàng)建一個(gè)
FnPtr
的向量并傳遞給eval_expression_with_scope
實(shí)現(xiàn)的。但這種方法比較繁瑣,并且只適用于簡(jiǎn)單的函數(shù)簽名。 - 函數(shù)名處理:函數(shù)名是通過(guò)字符串格式化直接嵌入到表達(dá)式中的,這意味著你需要確保傳入的函數(shù)名是安全的(即不會(huì)導(dǎo)致 Rhai 執(zhí)行不安全的代碼)。
一個(gè)更健壯的方法是使用 Rhai 的 FnCall
功能,但這需要更多的設(shè)置和錯(cuò)誤處理。
這是一個(gè)更通用的方法,但稍微復(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)建一個(gè) Rhai 引擎實(shí)例 let mut engine = Engine::new(); // 定義一個(gè) 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); // 將模塊注冊(cè)到引擎中 engine.register_module(module)?; // 創(chuàng)建一個(gè)作用域 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(); // 定義一個(gè)臨時(shí)的 Rhai 函數(shù)來(lái)調(diào)用目標(biā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(()) }
在這個(gè)更通用的示例中,我們定義了一個(gè) CallArgs
結(jié)構(gòu)體來(lái)存儲(chǔ)函數(shù)名和參數(shù),然后構(gòu)建了一個(gè)臨時(shí)的 Rhai 腳本,該腳本根據(jù)函數(shù)名和參數(shù)數(shù)量調(diào)用相應(yīng)的 Rhai 函數(shù)。
這種方法提供了更大的靈活性,但也更復(fù)雜,并且需要處理更多的錯(cuò)誤情況。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
利用Rust實(shí)現(xiàn)一個(gè)簡(jiǎn)單的Ping應(yīng)用
這兩年Rust火的一塌糊涂,甚至都燒到了前端,再不學(xué)習(xí)怕是要落伍了。最近翻了翻文檔,寫(xiě)了個(gè)簡(jiǎn)單的Ping應(yīng)用練練手,感興趣的小伙伴可以了解一下2022-12-12Rust語(yǔ)言的新手了解和學(xué)習(xí)入門(mén)啟蒙教程
這篇文章主要介紹了rust的特點(diǎn)、安裝、項(xiàng)目結(jié)構(gòu)、IDE環(huán)境配置、代碼運(yùn)行,講解了如何安裝Rust編譯器,創(chuàng)建和運(yùn)行第一個(gè)Rust程序,并對(duì)Rust語(yǔ)言的特點(diǎn)和優(yōu)勢(shì)作了說(shuō)明,包括內(nèi)存安全、高效性能、并發(fā)性、社區(qū)支持和統(tǒng)一包管理等,是新手了解和學(xué)習(xí)Rust語(yǔ)言的啟蒙教程2024-12-12Rust實(shí)現(xiàn)構(gòu)建器模式和如何使用Bon庫(kù)中的構(gòu)建器
這篇文章主要介紹了Rust實(shí)現(xiàn)構(gòu)建器模式和如何使用Bon庫(kù)中的構(gòu)建器,本文給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧2024-08-08Rust開(kāi)發(fā)WebAssembly在Html和Vue中的應(yīng)用小結(jié)(推薦)
這篇文章主要介紹了Rust開(kāi)發(fā)WebAssembly在Html和Vue中的應(yīng)用,本文將帶領(lǐng)大家在普通html上和vue手腳架上都來(lái)運(yùn)行wasm的流程,需要的朋友可以參考下2022-08-08Rust在寫(xiě)庫(kù)時(shí)實(shí)現(xiàn)緩存的操作方法
Moka是一個(gè)用于Rust的高性能緩存庫(kù),它提供了多種類型的緩存數(shù)據(jù)結(jié)構(gòu),包括哈希表、LRU(最近最少使用)緩存和?支持TTL(生存時(shí)間)緩存,這篇文章給大家介紹Rust在寫(xiě)庫(kù)時(shí)實(shí)現(xiàn)緩存的相關(guān)知識(shí),感興趣的朋友一起看看吧2024-01-01Rust中類型轉(zhuǎn)換在錯(cuò)誤處理中的應(yīng)用小結(jié)
隨著項(xiàng)目的進(jìn)展,關(guān)于Rust的故事又翻開(kāi)了新的一頁(yè),今天來(lái)到了服務(wù)器端的開(kāi)發(fā)場(chǎng)景,發(fā)現(xiàn)錯(cuò)誤處理中的錯(cuò)誤類型轉(zhuǎn)換有必要分享一下,對(duì)Rust錯(cuò)誤處理相關(guān)知識(shí)感興趣的朋友一起看看吧2023-09-09rust的nutyp驗(yàn)證和validator驗(yàn)證數(shù)據(jù)的方法示例詳解
本文介紹了在Rust語(yǔ)言中,如何使用nuType和validator兩種工具來(lái)對(duì)Cargo.toml和modules.rs文件進(jìn)行驗(yàn)證,通過(guò)具體的代碼示例和操作步驟,詳細(xì)解釋了驗(yàn)證過(guò)程和相關(guān)配置,幫助讀者更好地理解和掌握使用這兩種驗(yàn)證工具的方法,更多Rust相關(guān)技術(shù)資訊,可繼續(xù)關(guān)注腳本之家2024-09-09