
各位前端的老伙计、产品大大、运营伙伴们,大家好。我是写小程序架构写到能背出wxml编译器名字的老工程狮 👋
聊个真实场景——上个月帮一家做生鲜配送的团队看性能问题,技术负责人指着控制台跟我说:"哥,我把页面数据改了啊,为啥界面没反应?我明明在JS里把array[0].done改成true了。"
我看了眼代码就笑了。他还在用写Vue/react那种"直接改数据→期望框架自动发现"的心智模型,但setData没调,数据永远传不到另一边。他卡住的不是语法,是整个框架世界观:小程序不是单线程的"JS一手遮天",它是真真切切劈成两半在跑的——一半画界面,一半算逻辑,中间隔着一道Native桥。
今天咱就顺着官方那句干瘪的说明,把它掰开揉碎讲透:
整个小程序框架系统分为两部分:逻辑层(App Service)和视图层(View)。小程序提供了自己的视图层描述语言 WXML 和 WXSS,以及基于 JavaScript 的逻辑层框架,并在视图层与逻辑层间提供了数据传输和事件系统,让开发者能够专注于数据与逻辑。
一、先忘掉"前端是一整块"这件事
(1)你脑子里那个"前端"长啥样?
普通网页开发中,HTML/CSS/JS仨货全挤在同一个浏览器页面里。JS用document.querySelector捞节点→改style→界面变。数据是你的变量,界面是DOM树,俩人住同一间卧室,伸手就能摸到。
小程序把这套拆了。
渲染层(View)和逻辑层(App Service)不共享内存、不住一个线程。View负责"长什么样",App Service负责"算什么"。你想让界面变,就得走官方留给你的唯一正规通道——setData。想让逻辑层知道用户点了啥,就得走事件系统。没有捷径,也没有DOM给你抓。
(2)为什么说这是"保护",不是"限制"?
我早年也嫌它烦。但后来做过一个大促项目就明白了:当你的JS跑在一个纯JS引擎(JSCore一类)里,跟渲染完全隔离,意味着逻辑死循环最多卡逻辑,不一定把整个界面冻死。而且微信客户端(Native)在中转通信时能做安全检查、权限管控、请求过滤——这些东西如果全丢在一个裸露的浏览器线程里,你连防的地方都找不到。
二、视图层(View):WXML + WXSS + 组件,就是全部门面
(1)WXML不是HTML,但它故意长得很像
WXML(WeiXin Markup Language)是一套标签语言,用来描述页面结构。你看它长得像HTML,但它能用的标签是限定集合(view / text / image / scroll-view…),外加一套自己的"指令语法":
{{ }}数据绑定wx:if/wx:elif/wx:else条件wx:for列表渲染wx:key节点标识
举个最经典的例子(官方版的原味写法):
xml
<!-- pages/index/index.wxml --><viewclass="hello">{{name}}</view><buttonbindtap="changeName">点我</button>
// pages/index/index.jsPage({data: { name: 'Weixin' },changeName() {// 关键就在这:你不是去"改界面",你是去"改数据"this.setData({ name: 'MINA' })}})
你告诉框架:name在data里,{{name}}绑在视图上。之后所有事交出去——逻辑层算完调setData,框架自己决定哪儿要重绘、怎么把新数据送到渲染线程去。
(2)WXSS:CSS的堂弟,带一把rpx尺
WXSS核心就两点:大部分CSS能用,但加了rpx响应式单位 + 全局/局部作用域规则。
rpx的本质前文咱专门聊过——屏幕宽固定当750份分,你写750rpx就是满宽。它解决的最痛问题是:不用你手动算px × (设备宽/375)的适配系数,框架底层替你乘。
进阶建议(踩坑经验):
布局/间距/字号 → 放心用rpx
1px细边框 → 别指望
1rpx永远完美,改用1px或border + transform: scale兜底样式作用域 →
app.wxss管全局,页面.wxss管自己,组件有独立隔离。别用标签选择器乱轰全局
三、逻辑层(App Service):没有window的纯JS大脑
(1)它跑在哪?——不是浏览器
官方说得斩钉截铁:
小程序框架的逻辑层并非运行在浏览器中,因此 JavaScript 在 web 中一些能力都无法使用,如
window,document等。
它跑在JavaScript引擎(iOS常见JSCore环境)里,打包成一份JS文件,小程序生命周期内常驻,直到销毁——行为类似ServiceWorker,所以才叫App Service。
这意味着什么?
❌
document.getElementById→ undefined,连对象都没有❌
window.location.href = ...→ 没location给你跳✅
this.setData({...})→ 正道的数据下发✅
wx.navigateTo / switchTab→ 正道的路由✅
wx.request / wx.login / wx.pay→ 正道的原生能力调用
(2)App() 和 Page():框架要你"注册",不是"随便写"
小程序不是你new个Vue实例然后mount的那种风格。它是你按它的槽位填东西:
app.js → App({...}):全局入口,onLaunch/onShow/onHide管生命周期,globalData放全局共享数据
page.js → Page({ data, onLoad, ... }):每页的剧本,data是该页真相源,事件处理函数写这儿
老工程狮唠叨两句:
别把globalData当万能储物柜。它最适合放"登录态/环境配置/用户脱敏信息"。业务数据该在页面data就在页面data,该在组件就在组件。你越克制,三个月后回来改bug时越不容易骂自己。
四、数据传输 + 事件系统:这两条管道怎么转起来的
这部分是整个框架最容易被"以为懂了但其实没懂"的地方。
(1)下行:逻辑层 → 视图层,唯一正规通道是setData
你调setData({ name: 'MINA' }),发生的事大致是:
逻辑层把新data对象和旧对象做diff(虚拟DOM层面)
把需要传的部分序列化成字符串
跨线程/进程发给Native
Native转给渲染层
渲染层拿到后反序列化→更新虚拟DOM→刷真实渲染
因为逻辑层和渲染层不住一个世界,每一步都有代价:序列化、传输、反序列化、队列等待。所以核心铁律就一句:
setData只传渲染需要的东西,量越小越好,频率越低越好。
进阶建议(屡试不爽的写法):
js
// ❌ 别这样——把整个对象全撸过去this.setData({ wholePageData: this.data.wholePageData })// ✅ 这样——只改变的字段,最好用路径写法this.setData({ 'list[0].done': true })
渲染无关的变量(定时器句柄、缓存key、纯业务标记)直接挂this.xxx,别塞进data,不然白白增大每次传输包体。
(2)上行:视图层 → 逻辑层,靠事件系统
用户在View层点了按钮,这事怎么让App Service知道?
WXML里绑定:
xml
<buttonbindtap="handleBuy"data-sku="10086">立即购买</button>框架把这次触摸封装成一个事件对象e,通过Native中转,送给逻辑层你注册的handleBuy(e)。你要的信息从e.currentTarget.dataset.sku里拿——而不是去"查DOM"拿。
冒泡/阻止这档事咱之前聊事件时说过:想让它停住不往上冒,用catchtap别用bindtap。 点支付按钮却不慎触发了卡片跳转那种事故,本质就是冒泡没管住。
五、为什么这套"拆法"值得你认真理解(而不只是背文档)
(1)性能账
双线程+中转的设计,让小程序能做到:即便逻辑层在算重活,渲染层也不一定跟着窒息(当然你如果setData飙太快照样卡)。而且微信能把一些组件(map/video等)做成更接近原生的渲染,不在你的WebView里挤着画。
(2)合规账
把DOM从JS手里拿走,不只是架构口味问题——它还意味着你没法在逻辑层偷偷摸摸操作界面节点来绕过审核或做灰色注入。所有界面变化走声明式绑定+官方组件体系,行为更可审计。结合《个人信息保护法》里的最小必要原则,你在设计"哪些数据往下发、哪些字段脱敏后再发"时,天然多一层约束意识。
(3)团队协作账
当View和App Service的职责边界清晰:WXML/WXSS管长相,JS管逻辑,setData当桥,事件当信使——新人接手项目不用满文件搜document,直接看data+生命周期就知道页面什么状态、什么时机能干嘛。
六、结束语
回头读官方那句话,就不像一句干巴巴的定义了:
让开发者能够专注于数据与逻辑。
它不是在吹牛,而是在交代前提:因为View和App Service拆开、通信走管道、你得按注册规矩来——你的自由度被框住了,但换回来的是一套能跑在十亿台手机上、跟微信原生能力接驳、且不至于自己把自己绕成面条的秩序。
你越早停止"想抓DOM",越早开始"管数据",小程序写起来就越顺。
参考文献
[1] 微信团队. 框架(MINA)—— 响应的数据绑定与框架概述 [EB/OL]. (2024-05-01)[2024-06-16]. https://developers.weixin.qq.com/miniprogram/dev/framework/MINA.html.
[2] 微信团队. 逻辑层 App Service [EB/OL]. (2024-05-01)[2024-06-16]. https://developers.weixin.qq.com/miniprogram/dev/framework/app-service/index.html.
[3] 微信团队. WXML 语法参考 / 事件系统 [EB/OL]. (2024-05-01)[2024-06-16]. https://developers.weixin.qq.com/miniprogram/dev/reference/wxml/.
[4] GB/T 25000.51-2016, 系统与软件工程 系统与软件质量要求和评价(SQuaRE) 第51部分:就绪可用软件产品(RUSP)的质量要求和测试细则 [S]. 北京:中国标准出版社,2017.
💬 互动话题
你当初从网页转小程序时,最让你"卡住"的那个moment是啥?是想写document发现没有,还是数据改了但setData忘了调?评论区留一句你自己的"拆脑壳"故事——我挑几个最典型的,下篇顺着这条管道给你画一张完整的数据流转时序图(含Native中转点),帮你把那层雾彻底捅破。要《小程序双线程通信自查清单》的也喊一声~👇
夜雨聆风