当前位置: 首页 > news >正文

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

请添加图片描述

“如果结果不如你所愿,就在尘埃落定前奋力一搏。”——《夏目友人帐》
“有些事不是看到了希望才去坚持,而是因为坚持才会看到希望。”——《十宗罪》
“维持现状意味着空耗你的努力和生命。”——纪伯伦

Tauri 技术教程 * 第五章 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’;

  • 全局命令获取

    1. 在tauri.conf.json 中配置 app.withGlobalTauri 为true;
    2. 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教程-进阶篇-第二节 命令机制

“如果结果不如你所愿&#xff0c;就在尘埃落定前奋力一搏。”——《夏目友人帐》 “有些事不是看到了希望才去坚持&#xff0c;而是因为坚持才会看到希望。”——《十宗罪》 “维持现状意味着空耗你的努力和生命。”——纪伯伦 Tauri 技术教程 * 第五章 Tauri的进阶教程 第二节…...

candb++ windows11运行报错,找不到mfc140.dll

解决问题记录 mfc140.dll下载 注意&#xff1a;放置位置别搞错了...

提供的 IP 地址 10.0.0.5 和子网掩码位 /26 来计算相关的网络信息

网络和IP地址计算器 https://www.sojson.com/convert/subnetmask.html提供的 IP 地址 10.0.0.5 和子网掩码位 /26 来计算相关的网络信息。 子网掩码转换 子网掩码 /26 的含义二进制表示:/26 表示前 26 位是网络部分&#xff0c;剩下的 6 位是主机部分。对应的子网掩码为 255…...

vscode离线安装插件--终极解决方案

目录 离线安装插件安装方法 vscode连接远程服务器中的docker远程连接python jupyter开发 离线安装插件 使用vscode开发过程中&#xff0c;有一些内网服务器没法连接外网&#xff0c;造成安装插件不方便&#xff0c;网络上很多文章提供了很多方法&#xff0c;比较常见的一种是&…...

LabVIEW启动时Access Violation 0xC0000005错误

问题描述 在启动LabVIEW时&#xff0c;可能出现程序崩溃并提示以下错误&#xff1a;Error 0xC0000005 (Access Violation) ​ Access Violation错误通常是由于权限不足、文件冲突或驱动问题引起的。以下是解决此问题的全面优化方案&#xff1a; 解决步骤 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)文件传送协议

前言 一、文件传送协议&#xff08;FTP&#xff09; 概述&#xff1a; FTP&#xff08;File Transfer Protocol&#xff09;是互联网上使用得最广泛的文件传送协议。FTP提供交互式的访问&#xff0c;允许客户指明文件的类型与格式&#xff08;如指明是否使用ASCII码&#xff0…...

C++ STL之容器介绍(vector、list、set、map)

1 STL基本概念 C有两大思想&#xff0c;面向对象和泛型编程。泛型编程指编写代码时不必指定具体的数据类型&#xff0c;而是使用模板来代替实际类型&#xff0c;这样编写的函数或类可以在之后应用于各种数据类型。而STL就是C泛型编程的一个杰出例子。STL&#xff08;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学算法&#xff0c;本系列博客仅做个人记录&#xff0c;建议大家都去看carl本人的博客&#xff0c;写的真的很好的&#xff01; 代码随想录 LeetCode:131. 分割回文串 给你一个字符串 s&#xff0c;请你将 s 分割成一些子串&#xff0c;使每个子串都是回文串。返回 s 所…...

React-useState讲解

useState 让页面“动”起来 例如实现一个 click 计数功能&#xff0c;普通变量无法实现。即&#xff1a;修改普通变量无法触发组件的更新 rerender 通过 useState 即可实现。 state 是什么 State, A component’s memory —— 这个比喻非常好&#xff01; 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 题目 实现一个登录页面、实现一个注册页面&#xff1b;实现一个主页 - 登录页面&#xff1a;login.html - 注册页面&#xff1a;register.html - 主页…...

CentOS 6.8 安装 Nginx

个人博客地址&#xff1a;CentOS 6.8 安装 Nginx | 一张假钞的真实世界 提前安装&#xff1a; # sudo yum install yum-utils 一般情况下这个工具系统已经安装。 创建文件/etc/yum.repos.d/nginx.repo&#xff0c;输入内容如下&#xff1a; [nginx-stable] namenginx stab…...

网络网络层ICMP协议

网络网络层ICMP协议 1. ICMP 协议介绍 ICMP&#xff08;Internet Control Message Protocol&#xff09;是 TCP/IP 协议簇中的网络层控制报文协议。用于在 IP 主机、路由器之间传递控制消息&#xff0c;提供可能有关通信问题的反馈信息。 以及用于网络诊断或调试&#xff08;…...

当父级元素设置了flex 布局 ,两个子元素都设置了flex :1, 但是当子元素放不下的时候会溢出父元素怎么解决 (css 样式问题)

一、问题 遇到个样式问题&#xff0c;当父级元素设置了flex 布局 &#xff0c;两个子元素都设置了flex :1, 但是当子元素放不下的时候会溢出父元素怎么解决 &#xff08;拖拽浏览器 使页面变小&#xff09; 二、解决方法 .father{min-height: 600px;width: 100%;display: flex…...

Vue.js组件开发-如何实现路由懒加载

在Vue.js应用中&#xff0c;路由懒加载是一种优化性能的技术&#xff0c;它允许在需要时才加载特定的路由组件&#xff0c;而不是在应用启动时加载所有组件。这样可以显著减少初始加载时间&#xff0c;提高用户体验。在Vue Router中&#xff0c;实现路由懒加载非常简单&#xf…...

灵活妙想学数学

灵活妙想学数学 题1&#xff1a;海星有几只&#xff1f; 一共有12只海洋生物&#xff0c;分别是5只脚的海星&#xff0c;8只脚的章鱼和10只脚的鱿鱼&#xff0c;这些海洋动物的脚一共有87只&#xff0c;每种生物至少有1只&#xff0c;问海星有几只&#xff1f; 解&#xff1a…...

使用 Multer 上传图片到阿里云 OSS的两种方式

文件上传到哪里更好&#xff1f; 上传到服务器本地 上传到服务器本地&#xff0c;这种方法在现今商业项目中&#xff0c;几乎已经见不到了。因为服务器带宽&#xff0c;磁盘 IO 都是非常有限的。将文件上传和读取放在自己服务器上&#xff0c;并不是明智的选择。 上传到云储存…...

stm32G473的flash模式是单bank还是双bank?

今天突然有人stm32G473的flash模式是单bank还是双bank&#xff1f;由于时间太久&#xff0c;我真忘记了。搜搜发现&#xff0c;还真有人和我一样。见下面的链接&#xff1a;https://shequ.stmicroelectronics.cn/forum.php?modviewthread&tid644563 根据STM32G4系列参考手…...

IGP(Interior Gateway Protocol,内部网关协议)

IGP&#xff08;Interior Gateway Protocol&#xff0c;内部网关协议&#xff09; 是一种用于在一个自治系统&#xff08;AS&#xff09;内部传递路由信息的路由协议&#xff0c;主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...

解锁数据库简洁之道:FastAPI与SQLModel实战指南

在构建现代Web应用程序时&#xff0c;与数据库的交互无疑是核心环节。虽然传统的数据库操作方式&#xff08;如直接编写SQL语句与psycopg2交互&#xff09;赋予了我们精细的控制权&#xff0c;但在面对日益复杂的业务逻辑和快速迭代的需求时&#xff0c;这种方式的开发效率和可…...

C# 类和继承(抽象类)

抽象类 抽象类是指设计为被继承的类。抽象类只能被用作其他类的基类。 不能创建抽象类的实例。抽象类使用abstract修饰符声明。 抽象类可以包含抽象成员或普通的非抽象成员。抽象类的成员可以是抽象成员和普通带 实现的成员的任意组合。抽象类自己可以派生自另一个抽象类。例…...

【Go语言基础【13】】函数、闭包、方法

文章目录 零、概述一、函数基础1、函数基础概念2、参数传递机制3、返回值特性3.1. 多返回值3.2. 命名返回值3.3. 错误处理 二、函数类型与高阶函数1. 函数类型定义2. 高阶函数&#xff08;函数作为参数、返回值&#xff09; 三、匿名函数与闭包1. 匿名函数&#xff08;Lambda函…...

虚拟电厂发展三大趋势:市场化、技术主导、车网互联

市场化&#xff1a;从政策驱动到多元盈利 政策全面赋能 2025年4月&#xff0c;国家发改委、能源局发布《关于加快推进虚拟电厂发展的指导意见》&#xff0c;首次明确虚拟电厂为“独立市场主体”&#xff0c;提出硬性目标&#xff1a;2027年全国调节能力≥2000万千瓦&#xff0…...

pgsql:还原数据库后出现重复序列导致“more than one owned sequence found“报错问题的解决

问题&#xff1a; pgsql数据库通过备份数据库文件进行还原时&#xff0c;如果表中有自增序列&#xff0c;还原后可能会出现重复的序列&#xff0c;此时若向表中插入新行时会出现“more than one owned sequence found”的报错提示。 点击菜单“其它”-》“序列”&#xff0c;…...

C++--string的模拟实现

一,引言 string的模拟实现是只对string对象中给的主要功能经行模拟实现&#xff0c;其目的是加强对string的底层了解&#xff0c;以便于在以后的学习或者工作中更加熟练的使用string。本文中的代码仅供参考并不唯一。 二,默认成员函数 string主要有三个成员变量&#xff0c;…...

基于Python的气象数据分析及可视化研究

目录 一.&#x1f981;前言二.&#x1f981;开源代码与组件使用情况说明三.&#x1f981;核心功能1. ✅算法设计2. ✅PyEcharts库3. ✅Flask框架4. ✅爬虫5. ✅部署项目 四.&#x1f981;演示效果1. 管理员模块1.1 用户管理 2. 用户模块2.1 登录系统2.2 查看实时数据2.3 查看天…...

华为云Flexus+DeepSeek征文 | 基于Dify构建具备联网搜索能力的知识库问答助手

华为云FlexusDeepSeek征文 | 基于Dify构建具备联网搜索能力的知识库问答助手 一、构建知识库问答助手引言二、构建知识库问答助手环境2.1 基于FlexusX实例的Dify平台2.2 基于MaaS的模型API商用服务 三、构建知识库问答助手实战3.1 配置Dify环境3.2 创建知识库问答助手3.3 使用知…...