返回文章列表

一套代码四个平台

10 分钟
技术实践工具推荐

Web 应用多端适配技术实践笔记

事情的起因

前段时间,我给自家小朋友做了一个专属的小应用,Web 版的,里面有三个板块:玩游戏、培养习惯、听故事

最初是用 React 写的——React 是一个前端开发框架,你可以简单理解为"一套搭建网页的工具箱"。做出来之后,在浏览器里打开就能用,手机上也能访问,样式也做了适配。

本来以为这就完事了。

但用了几天发现一个很烦的问题:每次在手机浏览器里打开,顶部和底部的浏览器工具栏总会遮挡一部分内容。小朋友点来点去,有时候会误触到浏览器的前进后退按钮,体验很不好。

于是我就想:能不能把它变成一个"正经的 App"?不需要打开浏览器,直接点图标就能用的那种。

这一折腾,就折腾出了好几个方向。


一、先搞桌面端:用 Electron 把网页变成桌面应用

我最先尝试的是把这个 Web 应用做成桌面端应用,就是你在电脑上双击图标就能打开的那种软件。

用到的技术叫 Electron

Electron 是什么?

你可以这么理解:Electron 做的事情,就是把一个浏览器"藏"进了你的应用里。用户看到的是一个独立的桌面软件,但背后其实还是在跑网页代码。

很多你熟悉的软件其实都是用 Electron 做的,比如 VS Code(微软的代码编辑器)、Slack、Notion 桌面版。所以这个方案是很成熟的。

实际操作体验

这次转换的过程出乎意料地顺利。

我只是跟 Codex(OpenAI 的 AI 编程助手)说了一句"帮我把这个项目转换成 Electron 桌面应用",它就自动帮我完成了所有配置。用的是 electron-builder 这个工具来打包,一条命令就能把 React 项目打包成完整的桌面应用。

我先构建了 Windows 版本,成功了;又试了试 macOS 版本,也成功了。整个过程几乎没有手动写什么代码,Codex 把脏活累活都干了。

打包出来的应用体积确实比较大,随随便便就上百兆——因为 Electron 内置了一整个 Chromium 浏览器内核。对于我这个小应用来说,属实有点"杀鸡用牛刀"了,但桌面端嘛,大一点倒也能接受。


二、再搞移动端:用 Capacitor 把网页变成手机 App

桌面端搞定之后,我又想:小朋友日常用的是 iPad 和手机,能不能做成原生的移动端 App?

这次用到的框架叫 Capacitor

Capacitor 是什么?

简单来说,Capacitor 的思路和 Electron 有点像——它把你的网页代码塞进了手机系统自带的 WebView 里

WebView 是什么?你可以理解为手机系统内置的一个"迷你浏览器组件"。很多 App 里打开一个链接,不会跳转到外部浏览器,而是直接在 App 内部展示网页,用的就是 WebView。

Capacitor 做的事情,就是把你的 Web 应用包装成一个原生 App 的壳子,里面用 WebView 来渲染你的网页内容。好处是:一套 Web 代码,稍微配置一下,就能生成安卓的 APK 安装包

实际操作体验

和 Electron 一样,这次转换也是靠 Codex 完成的。

我跟它说"帮我把这个 Web 项目用 Capacitor 转成安卓应用",它就推荐了 Capacitor 这个方案,然后自动帮我完成了所有配置和构建。

装到手机上一试——还真能用,效果比我预期的要好。没有了浏览器工具栏的遮挡,全屏展示,小朋友操作起来顺畅多了。

而且打包出来的体积比 Electron 小很多,毕竟 WebView 是系统自带的,不需要额外打包一个浏览器进去。


三、再试试小程序:用 Taro 把 React 项目转成微信小程序

既然桌面端和移动端都搞定了,我又冒出一个念头:要不顺便做个微信小程序?毕竟在国内,小程序的使用场景太普遍了,分享起来也方便。

这里用到的框架叫 Taro

Taro 是什么?

Taro 是一个多端开发框架,它的卖点是:你用 React 的语法写一套代码,它帮你编译成微信小程序、支付宝小程序、H5 等不同平台的代码。听起来很美好对吧?

实际体验

说实话,跑通是跑通了——我确实成功把 React 项目通过 Taro 转成了微信小程序项目,在微信开发者工具里也能正常打开。

但是,样式偏差比较大。

按钮的位置不对、间距不对、有些组件的表现和 Web 端差异明显。背后的原因是 React 生态和小程序生态在样式处理上有不少差异,很多 CSS 属性在小程序里的表现不一样,需要单独去调。

总之这条路能走通,但没有 Capacitor 和 Electron 那么丝滑,需要额外花不少时间做适配和微调。


四、冷静下来想想:我真的需要这么多端吗?

折腾完上面这一大圈,我突然意识到一件事——

我给小朋友做一个应用,真的需要覆盖这么多平台吗?

仔细想想,我实际需要的其实就三个端:

  1. 手机端(日常随手用)
  2. iPad 端(在家里用,屏幕大,体验好)
  3. 网页端(偶尔在电脑上访问)

小程序暂时不急,桌面端也不是刚需。那些多出来的端侧方案,更多是我出于好奇心去做的技术探索。

这就是典型的"技术先行,需求在后"。

不过话说回来,我并不觉得这些探索是浪费时间。正是因为亲手试了一遍,我才真正理解了每个方案的优缺点和适用场景。下次再遇到类似需求的时候,我能很快做出判断,而不是在各种方案之间纠结。


五、下一步计划:用 Monorepo 统一管理多端代码

折腾完这一圈,我还发现了一个很现实的问题——我现在有四个代码仓库

  • React Web 项目(网页版)
  • Electron 项目(桌面版)
  • Capacitor 项目(安卓版)
  • Taro 项目(小程序版)

每次改一个功能,我得在四个仓库里分别改一遍。哪怕核心逻辑是一样的,也得复制粘贴四次。时间一长,各个仓库之间的代码必然会出现不一致,维护成本越来越高。

查了一下,发现业界对这类问题有一个成熟的解决方案,叫 Monorepo(单体仓库)。

简单来说,就是把所有端侧的代码放在同一个仓库里管理,但内部按子工程拆分。公共的代码抽成共享模块,各端只保留自己独有的部分。这样改一个地方,所有端自动同步,不会出现到处复制、到处改的问题。

这个方案我还没开始做,打算下一步尝试搭建一下,到时候再单独写一篇记录实践过程。


六、技术方案对比总结

最后,把这次用到的几个技术方案做一个横向对比,方便大家快速了解它们的区别和适用场景。

Electron vs Capacitor

维度ElectronCapacitor
目标平台桌面端(Windows / macOS / Linux)移动端(iOS / Android)
技术原理自己内置了一个 Chromium 浏览器借用系统自带的 WebView
应用体积比较大(因为要带一整个浏览器)比较轻量(WebView 是系统现成的)
内存占用偏高更低,接近原生 App

简单记就是:做桌面端用 Electron,做移动端用 Capacitor。

Capacitor vs React Native

  • Capacitor 是 "Web First":你的界面本质上还是网页(HTML + CSS + JS),只不过跑在了手机的 WebView 里。你几乎可以 100% 复用已有的 Web 代码。
  • React Native 是 "Native First":虽然你写的是 JS 代码,但它最终会被转换成真正的原生组件来渲染。

选择思路:

  • 已经有一个 Web 项目,想最快速度搬到手机上 → Capacitor,改动最小
  • 对性能和原生体验要求更高,愿意多投入一些开发成本 → React Native

一句话总结

场景推荐方案
Web 项目 → 桌面应用Electron
Web 项目 → 移动端 App(快速迁移)Capacitor
追求原生移动端体验React Native
React 项目 → 小程序Taro

写在最后

回顾这一轮折腾,本质上就是一个 Web 开发者试图"一鱼多吃"的过程——用一套 Web 代码,尽可能多地覆盖不同平台

现代前端生态确实提供了很多这样的工具和框架,让这件事变得越来越可行。而且有了 Codex 这样的 AI 编程助手,很多繁琐的配置工作都可以交给它来完成,大大降低了上手门槛。

但我也真切体会到:能做和该做是两件事。

技术上能跑通四五个端,不代表你真的需要维护四五个端。回到需求本身,想清楚用户到底在哪里用、怎么用,再做技术选型,才是更务实的做法。

技术实践工具推荐