Tauri教程-进阶篇-第二节 命令机制

“如果结果不如你所愿,就在尘埃落定前奋力一搏。”——《夏目友人帐》
“有些事不是看到了希望才去坚持,而是因为坚持才会看到希望。”——《十宗罪》
“维持现状意味着空耗你的努力和生命。”——纪伯伦
第二节 命令机制
一. 概述
本章节中我们将了解在前端项目中,如何调用Rust端的代码。
Tauri 提供了一个简单但功能强大的command系统,用于从 Web 应用调用 Rust 函数。命令可以接受参数并返回值。它们还可以返回错误和async。
二. 命令机制
Tauri 提供了一个命令功能,用于以类型安全的方式访问 Rust 函数,以及一个动态的事件系统。本章节我们先了解下他们的概念,然后了解它的工作内容和方式。
1. 命令 Command
Tauri 提供了一个简单但功能强大的command系统,用于从 Web 应用调用 Rust 函数。命令可以接受参数并返回值。它们还可以返回错误和async。
2. 基本概念
1. 命令注释
定义的命令需要采用 注释#[tauri::command] 进行修饰。如下:
#[tauri::command]
fn my_custom_command() {println!("I was invoked from JavaScript!");
}
命令的构成由:
-
注释
-
Rust函数基本构成
这里要注意:命令名称必须全局唯一。
2. 构造函数注入
通过构建函数进行注入,tauri::generate_handler 如下:
tauri::generate_handler![my_custom_command]
3. Invoke 函数
invoke函数的获取方式有2种
-
api 方式获取
import { invoke } from ‘@tauri-apps/api/core’;
-
全局命令获取
- 在tauri.conf.json 中配置 app.withGlobalTauri 为true;
- const invoke = window.TAURI.core.invoke;
前端调用方式:
invoke('my_custom_command');
4. 参数定义
参数定义的方式如下:
#[tauri::command]
fn my_custom_command(invoke_message: String) {println!("I was invoked from JavaScript, with this message: {}", invoke_message);
}
调用方式如下:
invoke('my_custom_command', { invokeMessage: 'Hello!' });
这里注意下:前端参数的名称与后端参数的名称:
后端是invoke_message,前端是json 需要是:invokeMessage,如果你需要保持一致可以采用snake_case 属性进行指定,如下:
#[tauri::command(rename_all = "snake_case")]
fn my_custom_command(invoke_message: String) {}invoke('my_custom_command', { invoke_message: 'Hello!' });
5. 返回值定义
返回值的定义:实例如下,和普通的函数定义没有区别
#[tauri::command]
fn my_custom_command() -> String {"Hello from Rust!".into()
}
在前端处理时,invoke 函数返回的是一个 promise,使用方式如下:
invoke('my_custom_command').then((message) => console.log(message));
返回大批量数据或者文件流时,需要结合 tauri::ipc::Response 来使用,方式如下:
use tauri::ipc::Response;
#[tauri::command]
fn read_file() -> Response {let data = std::fs::read("/path/to/file").unwrap();tauri::ipc::Response::new(data)
}
3. 异常处理
在Tauri 编程中,异常或者错误的处理需要 实现serde::Serialize,在程序开发中我闷可以使用Result 来返回处理信息,也可以自定义错误来返回处理信息,下面我闷来演示下,如何使用这2种方式
-
Result 的处理
#[tauri::command] fn login(user: String, password: String) -> Result<String, String> {if user == "tauri" && password == "tauri" {// resolveOk("logged_in".to_string())} else {// rejectErr("invalid credentials".to_string())} } -
自定义错误类型,这里我们使用thiserror库辅助构建
#[derive(Debug, thiserror::Error)] enum Error {#[error(transparent)]Io(#[from] std::io::Error),#[error("failed to parse as string: {0}")]Utf8(#[from] std::str::Utf8Error), }#[derive(serde::Serialize)] #[serde(tag = "kind", content = "message")] #[serde(rename_all = "camelCase")] enum ErrorKind {Io(String),Utf8(String), }impl serde::Serialize for Error {fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>whereS: serde::ser::Serializer,{let error_message = self.to_string();let error_kind = match self {Self::Io(_) => ErrorKind::Io(error_message),Self::Utf8(_) => ErrorKind::Utf8(error_message),};error_kind.serialize(serializer)} }#[tauri::command] fn read() -> Result<Vec<u8>, Error> {let data = std::fs::read("/path/to/file")?;Ok(data) }此时我们在前端使用Invoke的catch 捕获时就会得到一个 { kind: ‘io’ | ‘utf8’, message: string }`错误对象:
type ErrorKind = {kind: 'io' | 'utf8';message: string; };invoke('read').catch((e: ErrorKind) => {});
4. 异步操作
异步操作也是我们在交互种普遍采用的一种方式,那么在tauri中我们怎么去实现它,
开启的方式比较简单,只需要在命令需要异步运行,将其声明为async,如下:
#[tauri::command]
async fn my_custom_command(value: String) -> String {.......
}
在 async 操作需要注意
异步命令使用 在单独的异步任务上执行,没有async 的命令将在主线程上执行
异步返回类型的定义:
在返回类型的定义时,推荐大家采用Result<a,b>的方式来进行处理;
Result<String, ()>返回一个字符串,并且没有错误。Result<(), ()>返回null。Result<bool, Error>返回布尔值或错误。
代码示例:
// Result<String, ()>
#[tauri::command]
async fn my_custom_command(value: &str) -> Result<String, ()> {some_async_function().await;Ok(format!(value))
}
前端的处理方式:没有什么区别
invoke('my_custom_command', { value: 'Hello, Async!' }).then(() =>console.log('Completed!')
);
5. 数据传输
在Tauri 项目开发中,我们会遇到流的操作,关于如何操作流,Tauri 提供了通道技术,用于应对流传输的数据机制。即: tauri::ipc::Channel;例如下载进度、子进程输出和 WebSocket 消息。
使用起来也是比较简单的,下面我们来看一个下载的示例代码
在Rest中定义下载命令
use tauri::{AppHandle, ipc::Channel};
use serde::Serialize;#[derive(Clone, Serialize)]
#[serde(rename_all = "camelCase", tag = "event", content = "data")]
enum DownloadEvent<'a> {#[serde(rename_all = "camelCase")]Started {url: &'a str,download_id: usize,content_length: usize,},#[serde(rename_all = "camelCase")]Progress {download_id: usize,chunk_length: usize,},#[serde(rename_all = "camelCase")]Finished {download_id: usize,},
}#[tauri::command]
fn download(app: AppHandle, url: String, on_event: Channel<DownloadEvent>) {let content_length = 1000;let download_id = 1;on_event.send(DownloadEvent::Started {url: &url,download_id,content_length,}).unwrap();for chunk_length in [15, 150, 35, 500, 300] {on_event.send(DownloadEvent::Progress {download_id,chunk_length,}).unwrap();}on_event.send(DownloadEvent::Finished { download_id }).unwrap();
}
以上示例中我们定义了2个内容:
-
DownloadEvent
枚举类,里面包含了下载文件的信息(地址 大小 id),进度
-
download
接受2个参数,一个下载地址,一个事件处理
前端代码:
import { invoke, Channel } from '@tauri-apps/api/core';type DownloadEvent =| {event: 'started';data: {url: string;downloadId: number;contentLength: number;};}| {event: 'progress';data: {downloadId: number;chunkLength: number;};}| {event: 'finished';data: {downloadId: number;};};const onEvent = new Channel<DownloadEvent>();
onEvent.onmessage = (message) => {console.log(`got download event ${message.event}`);
};await invoke('download', {url: 'xxxxxx',onEvent,
});
6. 命令交互
在定义的命令中,我们可以访问 WebviewWindow AppHandle 状态 原始请求对象,下面我们来具体看下如何去使用他们
1. 访问 WebviewWindow
#[tauri::command]
async fn my_custom_command(webview_window: tauri::WebviewWindow) {println!("WebviewWindow: {}", webview_window.label());
}
2. 访问 WebviewWindow
#[tauri::command]
async fn my_custom_command(app_handle: tauri::AppHandle) {let app_dir = app_handle.path_resolver().app_dir();use tauri::GlobalShortcutManager;app_handle.global_shortcut_manager().register("CTRL + U", move || {});
}
3. 访问状态
状态我们会在后续的状态管理中详细了解它,这里我们先了解操作它的方式,在状态管理中我闷还会在进行探讨。
Tauri 通过 tauri::Builder.manage 来绑定和管理状态,使用时科通过tauri::State 处理绑定的状态即可
如下:
// 定义一个状态信息
struct MyState(String);#[tauri::command]
fn my_custom_command(state: tauri::State<MyState>) { // 使用assert_eq!(state.0 == "some state value", true);
}#[cfg_attr(mobile, tauri::mobile_entry_point)]
pub fn run() {tauri::Builder::default()// 绑定并管理.manage(MyState("some state value".into())).invoke_handler(tauri::generate_handler![my_custom_command]).run(tauri::generate_context!()).expect("error while running tauri application");
}
4. 访问 请求对象
在请求处理中,有时我们需要访问包含原始主体有效信息和请求标头的完整对象,可以通过tauri::ipc::Request对象进行处理
#[tauri::command]
fn upload(request: tauri::ipc::Request) -> Result<(), Error> {// request.body() request.headers()// upload...Ok(())
}
在前端,你可以调用invoke()来发送原始请求体,
__TAURI__.core.invoke('upload', {}, {headers: {Authorization: 'apikey',},
});
7. 命令抽取
在项目开发中,我们不可能将所有的命令都定义在主程序中,这样不利于开发管理,也不利于修改,通常我们会将命令单独提取为一个文件或者多个文件,在主程序中使用他们,以达到开发分工的情况,提升开发效率和维护便捷性。
-
定义一个command.rs 在 src-tauri src 目录下
#[tauri::command] fn cmd_a() -> String {"Command a".to_string() }#[tauri::command] fn cmd_b() -> String {"asdsa".to_string() }// Learn more about Tauri commands at https://tauri.app/develop/calling-rust/ #[tauri::command] pub fn greet(name: &str) -> String {format!("Hello, {}! You've been greeted from Rust!", name) } -
在main 中 配置
mod command;#[cfg_attr(mobile, tauri::mobile_entry_point)] pub fn run() {tauri::Builder::default().....invoke_handler(tauri::generate_handler![command::greet])..... }
相关文章:
Tauri教程-进阶篇-第二节 命令机制
“如果结果不如你所愿,就在尘埃落定前奋力一搏。”——《夏目友人帐》 “有些事不是看到了希望才去坚持,而是因为坚持才会看到希望。”——《十宗罪》 “维持现状意味着空耗你的努力和生命。”——纪伯伦 Tauri 技术教程 * 第五章 Tauri的进阶教程 第二节…...
candb++ windows11运行报错,找不到mfc140.dll
解决问题记录 mfc140.dll下载 注意:放置位置别搞错了...
提供的 IP 地址 10.0.0.5 和子网掩码位 /26 来计算相关的网络信息
网络和IP地址计算器 https://www.sojson.com/convert/subnetmask.html提供的 IP 地址 10.0.0.5 和子网掩码位 /26 来计算相关的网络信息。 子网掩码转换 子网掩码 /26 的含义二进制表示:/26 表示前 26 位是网络部分,剩下的 6 位是主机部分。对应的子网掩码为 255…...
vscode离线安装插件--终极解决方案
目录 离线安装插件安装方法 vscode连接远程服务器中的docker远程连接python jupyter开发 离线安装插件 使用vscode开发过程中,有一些内网服务器没法连接外网,造成安装插件不方便,网络上很多文章提供了很多方法,比较常见的一种是&…...
LabVIEW启动时Access Violation 0xC0000005错误
问题描述 在启动LabVIEW时,可能出现程序崩溃并提示以下错误:Error 0xC0000005 (Access Violation) Access Violation错误通常是由于权限不足、文件冲突或驱动问题引起的。以下是解决此问题的全面优化方案: 解决步骤 1. 以管理员身份运行…...
string(一)
一、了解string 可以看成是字符顺序表。 二、string遍历方式 1、下标[ ] 重载了[] for(int i 0; i < s.size(); i) {cout << s[i]; } 2、迭代器 auto it s.begin(); while(it ! s.end()) {cout << *it;it; } 3、范围for for(auto ch : s) {cout <&l…...
计算机网络 (41)文件传送协议
前言 一、文件传送协议(FTP) 概述: FTP(File Transfer Protocol)是互联网上使用得最广泛的文件传送协议。FTP提供交互式的访问,允许客户指明文件的类型与格式(如指明是否使用ASCII码࿰…...
C++ STL之容器介绍(vector、list、set、map)
1 STL基本概念 C有两大思想,面向对象和泛型编程。泛型编程指编写代码时不必指定具体的数据类型,而是使用模板来代替实际类型,这样编写的函数或类可以在之后应用于各种数据类型。而STL就是C泛型编程的一个杰出例子。STL(Standard …...
redisson 连接 redis5报错 ERR wrong number of arguments for ‘auth‘ command
依赖版本 org.redisson:redisson-spring-boot-starter:3.25.2 现象 启动报错 org.redisson.client.RedisException: ERR wrong number of arguments for ‘auth’ command. channel: [xxx] command: (AUTH), params: (password masked) 原因 redis6以下版本认证参数不包含用…...
LeetCode:131. 分割回文串
跟着carl学算法,本系列博客仅做个人记录,建议大家都去看carl本人的博客,写的真的很好的! 代码随想录 LeetCode:131. 分割回文串 给你一个字符串 s,请你将 s 分割成一些子串,使每个子串都是回文串。返回 s 所…...
React-useState讲解
useState 让页面“动”起来 例如实现一个 click 计数功能,普通变量无法实现。即:修改普通变量无法触发组件的更新 rerender 通过 useState 即可实现。 state 是什么 State, A component’s memory —— 这个比喻非常好! props 父组件传…...
混币器是什么,波卡跨链交易平台
混币器是什么 混币器是一种加密货币工具,主要功能是将用户的加密货币与其他众多用户的加密货币混合在一起,打乱资金的流向和交易痕迹,使得加密货币的来源和去向难以追踪,从而增加交易的匿名性和隐私性。以下是对其工作流程和相关举例的介绍: 工作流程 用户首先将自己的加…...
【PHP】双方接口通信校验服务
请求方 使用 ApiAuthService::buildUrl($domain, [terminal > 1, ts > time()]); //http://域名/adminapi/login/platformLogin?signF7FE8A150DEC18BE8A71C5059742C81A&terminal1&ts1736904841接收方 $getParams $this->request->get();$validate ApiA…...
Web第一次作业
目录 题目 html代码 index login register css代码 base index login register 效果展示 index login register 题目 实现一个登录页面、实现一个注册页面;实现一个主页 - 登录页面:login.html - 注册页面:register.html - 主页…...
CentOS 6.8 安装 Nginx
个人博客地址:CentOS 6.8 安装 Nginx | 一张假钞的真实世界 提前安装: # sudo yum install yum-utils 一般情况下这个工具系统已经安装。 创建文件/etc/yum.repos.d/nginx.repo,输入内容如下: [nginx-stable] namenginx stab…...
网络网络层ICMP协议
网络网络层ICMP协议 1. ICMP 协议介绍 ICMP(Internet Control Message Protocol)是 TCP/IP 协议簇中的网络层控制报文协议。用于在 IP 主机、路由器之间传递控制消息,提供可能有关通信问题的反馈信息。 以及用于网络诊断或调试(…...
当父级元素设置了flex 布局 ,两个子元素都设置了flex :1, 但是当子元素放不下的时候会溢出父元素怎么解决 (css 样式问题)
一、问题 遇到个样式问题,当父级元素设置了flex 布局 ,两个子元素都设置了flex :1, 但是当子元素放不下的时候会溢出父元素怎么解决 (拖拽浏览器 使页面变小) 二、解决方法 .father{min-height: 600px;width: 100%;display: flex…...
Vue.js组件开发-如何实现路由懒加载
在Vue.js应用中,路由懒加载是一种优化性能的技术,它允许在需要时才加载特定的路由组件,而不是在应用启动时加载所有组件。这样可以显著减少初始加载时间,提高用户体验。在Vue Router中,实现路由懒加载非常简单…...
灵活妙想学数学
灵活妙想学数学 题1:海星有几只? 一共有12只海洋生物,分别是5只脚的海星,8只脚的章鱼和10只脚的鱿鱼,这些海洋动物的脚一共有87只,每种生物至少有1只,问海星有几只? 解:…...
使用 Multer 上传图片到阿里云 OSS的两种方式
文件上传到哪里更好? 上传到服务器本地 上传到服务器本地,这种方法在现今商业项目中,几乎已经见不到了。因为服务器带宽,磁盘 IO 都是非常有限的。将文件上传和读取放在自己服务器上,并不是明智的选择。 上传到云储存…...
告别教材下载烦恼:国家中小学智慧教育平台电子课本解析工具如何实现3分钟高效获取
告别教材下载烦恼:国家中小学智慧教育平台电子课本解析工具如何实现3分钟高效获取 【免费下载链接】tchMaterial-parser 国家中小学智慧教育平台 电子课本下载工具,帮助您从智慧教育平台中获取电子课本的 PDF 文件网址并进行下载,让您更方便地…...
终极指南:如何用res-downloader一键下载全网无水印资源
终极指南:如何用res-downloader一键下载全网无水印资源 【免费下载链接】res-downloader 视频号、小程序、抖音、快手、小红书、直播流、m3u8、酷狗、QQ音乐等常见网络资源下载! 项目地址: https://gitcode.com/GitHub_Trending/re/res-downloader 你是否经常…...
LibreCAD:完全免费的2D CAD软件终极指南,告别昂贵许可证
LibreCAD:完全免费的2D CAD软件终极指南,告别昂贵许可证 【免费下载链接】LibreCAD LibreCAD is a cross-platform 2D CAD program written in C17. It can read DXF/DWG files and can write DXF/PDF/SVG files. It supports point/line/circle/ellipse…...
保姆级教程:在PVE 8.3上搞定Windows 11和Server 2025的VirtIO驱动安装与优化
PVE 8.3虚拟化环境下的Windows系统性能优化全攻略 在虚拟化技术日益普及的今天,Proxmox VE(PVE)作为开源的虚拟化平台,因其稳定性和灵活性受到众多技术爱好者和企业用户的青睐。然而,许多用户在PVE上部署Windows系统时…...
卡尔曼滤波调参实战:如何用MATLAB让MPU6050的加速度数据更‘听话’?
卡尔曼滤波调参实战:如何用MATLAB让MPU6050的加速度数据更‘听话’? 当你在MATLAB中第一次看到MPU6050的原始加速度数据时,那些疯狂跳动的曲线可能会让你怀疑人生。别担心,这不是传感器坏了,而是现实世界本就充满噪声…...
深度学习优化算法详解:从 SGD 到 AdamW
深度学习优化算法详解:从 SGD 到 AdamW 1. 背景与动机 优化算法是深度学习训练的核心,选择合适的优化器直接影响模型的收敛速度和最终性能。本文深入分析主流优化算法的原理和适用场景。 2. 梯度下降家族 2.1 SGD import torch import torch.nn as nnopt…...
Ollama实测:Yi-Coder-1.5B代码生成速度有多快?3秒搞定日常函数
Ollama实测:Yi-Coder-1.5B代码生成速度有多快?3秒搞定日常函数 1. 测试背景与目标 作为一名开发者,每天都要面对各种编码任务。从简单的工具函数到复杂的算法实现,代码生成速度直接影响着开发效率。Yi-Coder-1.5B作为一款开源的…...
KOReader 2025.04:重新定义电子墨水屏阅读
KOReader 2025.04:重新定义电子墨水屏阅读 【免费下载链接】koreader An ebook reader application supporting PDF, DjVu, EPUB, FB2 and many more formats, running on Cervantes, Kindle, Kobo, PocketBook and Android devices 项目地址: https://gitcode.co…...
MediaPipe Pose镜像体验:CPU也能毫秒级检测,无需GPU免配置
MediaPipe Pose镜像体验:CPU也能毫秒级检测,无需GPU免配置 1. 引言:CPU上的实时姿态检测革命 在计算机视觉应用中,人体姿态检测一直是个热门领域。从健身应用的动作分析到虚拟试衣的体型测量,这项技术正在改变我们与…...
cv_resnet101_face-detection_cvpr22papermogface保姆级教程:GPU显存占用监控与自动释放策略
cv_resnet101_face-detection_cvpr22papermogface保姆级教程:GPU显存占用监控与自动释放策略 1. 引言 如果你正在使用基于ResNet101的MogFace人脸检测模型,可能会遇到一个常见问题:GPU显存占用越来越高,最终导致程序崩溃。尤其是…...
