Flutter、Electron、Tauri框架对比及Electron+Vue项目实战
随着移动互联网和桌面应用的快速发展,开发者们越来越倾向于使用跨平台框架来进步开发效率和降低资本。Flutter、Electron和Tauri作为当前最受欢迎的几种跨平台框架,各自拥有独特的优势和应用场景。本文旨在深入探究这三种框架的特点、优缺点,并分享一个基于Electron+Vue的项目实战经验。
背景
XXXX团队:我们想在断网环境下通过 RuntimeAI 运行 Python 程序,给用户输出日志、执行效果等需求的桌面端程序。
我:好的,上菜。
三大框架对比
Flutter简介
Flutter是Google推出的开源UI软件开发工具包,用于构建在iOS、Android、Web和桌面端高质量的原生界面。Flutter的一大特点是其“一次编写,处处运行”的本事,极大地进步了开发效率。Flutter使用Dart语言,拥有丰富的组件库和精良的性能。
Electron简介
Electron是一个使用JavaScript、HTML和CSS构建桌面应用的框架,由GitHub开发。它允许开发者将Web技能用于桌面软件的开发,使得原本只能在网络欣赏器中运行的应用能够被作为独立的桌面应用来分发和安装。Electron应用本质上是Chromium欣赏器的变种,这意味着它相对较重,但开发速率快,易于上手。
Tauri简介
Tauri 是一个相对较新的框架,旨在提供一个安全、轻量级的平台,用于将Web技能包装成桌面应用。与Electron相比,Tauri的最大优势在于其对安全性的强调以及更小的应用体积。Tauri应用使用Rust语言进行系统调用,提供了更好的性能和更低的资源消耗。
特点FlutterElectronTauri开发语言DartJavaScript, HTML, CSSJavaScript, HTML, CSS跨平台移动、Web和桌面应用开发桌面应用开发桌面应用开发性能高(Flutter直接编译到机器代码,淘汰了桥接层)中(基于Chromium和Node.js,可能占用更多资源)高(天生的应用体积小,性能优化)用户界面丰富的UI组件库,可高度自定义使用Web技能构建UI,机动使用Web技能构建UI,机动应用体积小较大(包含Chromium和Node.js)中安全性依赖于框架自己和开发实践依赖于框架自己和开发实践,可能需要注意Web安全问题高(Rust的安全性优势)实用场景得当需要高性能和高度定制UI的跨平台应用得当希望快速将Web应用转为桌面应用的项目得当对应用体积和性能有严酷要求,注重安全的桌面应用开发社区支持强强弱GitHub Star161k111k75.5k 数据对比
根据三大框架官方文档,安装了开发和构建依赖、配置环境变量和乐成启动后,我将通过启动速率、目次结构、打包体积做真实对比。
统一开发IDE为 VS Code
- 启动速率:Electron > Tauri > Flutter
启动速率只做大概预估,因为跟机器性能、网速和Tauri是否有新的依赖需要更新有关。
- 目次结构:Electron 、Tauri > Flutter
- 打包体积:Flutter > Tauri > Electron
框架选择
基于以上对比数据,我选择Electron,来由是:
- 开发难度小
- 安装依赖方便
- 更加得当本次需求
- VS Code也是使用Electron开发
- 社区庞大
Electron+Vue3项目实战底子版)
如果安装 electron、electron-builder和打包出错。中央思想就是自己下载好所需依赖,放至对应目次下。
搭建Electron应用参考文章,部分需要做修改。
首先,按照 Vue 官网步骤创建一个Vite+Vue+Ts项目:
- npm create vue@latest
- cd my-project
- code .
- npm install
复制代码- npm install electron vite-plugin-electron -D
复制代码 dist-electron:vite-plugin-electron插件运行后,会将electron主进程文件编译到dist-electron/index.js
chcp 65001:解决中文乱码
去掉 “type”: “module”
- "main": "dist-electron/index.js"
- "scripts": {
- "dev": "chcp 65001 && vite"
- ...
- }
复制代码- import { fileURLToPath, URL } from 'node:url'
- import { defineConfig } from 'vite'
- import vue from '@vitejs/plugin-vue'
- import electron from 'vite-plugin-electron'
- export default defineConfig({
- plugins: [
- vue(),
- electron({
- entry: "electron/index.ts"
- })
- ]
- })
复制代码
- 创建 electron/index.ts 主进程文件
- import { app, BrowserWindow } from "electron";
- let win;
- const createWindow = () => {
- win = new BrowserWindow({
- webPreferences: {
- contextIsolation: false, // 允许渲染器脚本访问预加载脚本中的全局变量和模块
- nodeIntegration: true // 允许在渲染进程中使用node
- },
- title: "Main window",
- width: 800,
- height: 600
- });
- // 判断当前环境,加载不同html文件
- if (process.env.NODE_ENV != "development") {
- win.loadFile("dist/index.html");
- } else {
- win.loadURL(`${process.env["VITE_DEV_SERVER_URL"]}`);
- }
- };
- // electron资源加载完毕,创建窗口
- app.whenReady().then(createWindow);
- // 窗口关闭,退出electron
- app.on("window-all-closed", () => {
- app.quit();
- });
复制代码
- npm i electron-builder -D
复制代码
- package.json添加打包脚本和build配置
可以自定义应用的logo,详情参考官方文档
- "scripts": {
- "build": "vite build && electron-builder"
- },
- "build": {
- "appId": "electron.desktop",
- "productName": "vue-electron",
- "asar": true,
- "copyright": "Copyright © 2024",
- "directories": {
- // exe文件输出到release下
- "output": "release/"
- },
- "files": [
- "dist/**",
- "dist-electron/**"
- ],
- "win": {
- "target": [
- {
- "target": "nsis",
- "arch": [
- "x64"
- ]
- }
- ],
- "artifactName": "${productName}_${version}.${ext}"
- },
- "nsis": {
- "oneClick": false,
- "perMachine": false,
- "allowToChangeInstallationDirectory": true,
- "deleteAppDataOnUninstall": false
- },
- }
复制代码 运行命令后,会天生release文件夹,此中包含了可执行文件vue-electron_0.0.0.exe。安装即可
Electron+Vue3项目实战(进阶版)
给底子版添加新功能:前端点击按钮 --> 调用机器的Python程序 --> 执行自定义py脚本 --> 返回效果给前端
前提:机器已经乐成安装 Python3
- # 根目录下新增scripts/script.py
- print("hello world")
复制代码- import { app, BrowserWindow } from 'electron'
- import { spawn } from 'child_process'
- import path from 'path'
- let win;
- const createWindow = () => {
- win = new BrowserWindow({
- webPreferences: {
- contextIsolation: false,
- nodeIntegration: true,
- },
- title: "Main window",
- width: 800,
- height: 600
- })
- if(process.env.NODE_ENV != "development") {
- win.loadFile("dist/index.html")
- } else {
- win.loadURL(`${process.env["VITE_DEV_SERVER_URL"]}`)
- }
- }
- // start 创建node子进程,执行脚本
- const pythonProcess = spawn('python', [path.join(process.cwd(), 'scripts/script.py')])
- pythonProcess.stdout.on('data', res => {
- // res为buffer格式,需要转换字符格式
- console.log('success',res.toString());
- })
- pythonProcess.stderr.on('data', err => {
- console.log('err', err.toString());
- })
- // end
- app.whenReady().then(createWindow);
- app.on('window-all-closed', () => {
- app.quit()
- })
复制代码- build started...
- ✓ 1 modules transformed.
- dist-electron/index.js 0.95 kB │ gzip: 0.55 kB
- built in 33ms.
- success hello world
复制代码 主进程与渲染进程
- npm i vite-plugin-electron-renderer -D
复制代码- // vite.config.ts
- import electronRender from 'vite-plugin-electron-renderer'
- plugins: [
- vue(),
- electron({
- entry: "electron/index.ts"
- }),
- electronRender()
- ],
复制代码- <script setup lang="ts">
- import { ipcRenderer } from 'electron'
- // 添加点击事件,发送消息给主进程
- const runPy = () => {
- ipcRenderer.send('run')
- }
- </script>
- <template>
- <header>
- <img alt="Vue logo" class="logo" src="./assets/logo.svg" width="125" height="125" @click="runPy"/>
- </header>
- </template>
复制代码- // 首先将执行的子进程提取出来
- const run = () => {
- return new Promise((resolve, reject) => {
- const pythonProcess = spawn('python', [path.join(process.cwd(), 'scripts/script.py')])
- pythonProcess.stdout.on('data', res => {
- resolve(res.toString())
- })
- pythonProcess.stderr.on('data', err => {
- reject(err.toString())
- })
- })
- }
- // 主进程监听到渲染进程的消息后,立马执行run函数,执行结束后再通知渲染进程
- ipcMain.on('run', () => {
- run()
- .then(res => {
- win.webContents.send("finish", res)
- })
- .catch(err => {
- win.webContents.send("finish", err)
- })
- })
复制代码- <script setup lang="ts">
- import { ipcRenderer } from 'electron'
- import { ref } from 'vue'
- const runPy = () => {
- ipcRenderer.send('run')
- }
- const result = ref()
- // 监听主进程发送的finish消息。实际发送的数据在回调的第二个参数中
- ipcRenderer.on('finish', (_, data:any) => {
- result.value = data
- })
- </script>
- <template>
- <header>
- <img alt="Vue logo" class="logo" src="./assets/logo.svg" width="125" height="125" @click="runPy"/>
- <p>子进程执行Python后的结果:<b>{{ result }}</b></p>
- </header>
- </template>
复制代码
结语
- Electron、nodejs有大量api需要在实践中慢慢摸索,以上只是冰山一角
- 奇希奇怪的报错,大概率是node版本过低导致
- 打包后的Flutter可执行文件无法显示,但有对应的进程在跑。修改如下代码:
- // windows\runner\flutter_window.cpp
- // 只留下 this->Show();
- //flutter_controller_->engine()->SetNextFrameCallback([&]() {
- this->Show();
- //});
复制代码 免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |