官网:qiu8310.github.io/minapp/html
做者:Moravue
在原生小程序开发中,数据流是单向的,没法双向绑定,可是要实现双向绑定的功能仍是蛮简单的!git
下文要讲的是小程序框架 minapp 中实现双向绑定的原理,在 minapp 中,你只须要在 wxml 模板中给组件的属性名后加上
.sync
就能够实现双向绑定。下面为了解释其原理,过程可能会说的稍微复杂些,但其实 minapp 框架已经处理了那些繁杂的细节!github
首先,要使数据双向绑定,应该避免过多的数据源。 在数据从上到下天然流动的状况下,若是每一个组件中都维护它们本身的数据,而又要保持它们数据值的一致,这虽然能够作到,但实现过程并不会简单。 可是也不必说为了有一个统一的数据源就使用 mobx 或 redux 来全局管理数据,这就有点杀鸡用牛刀的感受了。 因为双向绑定只存在于父子组件之间,而数据又是从父到子传递的,因此能够优先使用父组件中的数据为数据源, 子组件每次更新数据并不更新它本身内部的数据,而是经过事件机制触发父组件更新它的数据,而父组件更新数据后又会将更新的数据天然地传给子组件, 由此达到数据的双向流动!redux
并非全部数据都须要双向绑定,也并非全部数据都是对外的,子组件还能够有它本身的一个内部数据。因此这就涉及到咱们要说的第二个问题:区分哪些数据须要双向绑定,哪些数据又须要子组件本身维护。小程序
用过 vue 的应该都知道,在 vue 中要实现双向绑定,须要在模板中作特殊处理。好比要将父组件的 parentAttr
双向绑定到子组件的 childAttr
上,须要在父组件的模板中这样写:微信小程序
<child childAttr.sync="parentAttr" />
复制代码
可是小程序并无这样的简单的语法,小程序的 wxml 语言的属性名中甚至都不容许出现 " . " 这样的字符。回到咱们的问题上来,子组件须要知道哪些属性须要双向绑定,哪些属性须要本身维护, 给模板加个字段(syncAttrMap
)专门来告诉子组件须要双向绑定的数据集合不就好了么。如,能够将上面的示例写成微信小程序支持的写法:微信
<child childAttr="{{parentAttr}}" syncAttrMap="childAttr=parentAttr" />
<!-- 若是同时存在多个双向绑定和不须要双向绑定的属性时,能够写成下面这样: p1, p2 分别双向绑定到子组件的 c1, c2,而 p3 单向绑定到 c3 上 -->
<child c1="{{p1}}" c2="{{p2}}" c3="{{p3}}" syncAttrMap="c1=p1&c2=p2" />
复制代码
接着,就须要处理子组件数据更新的问题了,在子组件中有两部分数据,一部分是内部数据,另外一部分是父组件中的数据, 子组件能够经过读取属性 syncAttrMap
来获得哪些数据是内部的数据,哪些数据是父组件的数据,而且能够知道对应 的父组件中的数据的键名是什么。因为原生的组件方法 setData
不会管你是内部数据,仍是父组件中的数据,只要 你调用它去更新数据,它只会更新内部的数据。因此须要另外实现一个新的方法,来自动判断数据源,若是是内部数据, 则直接调用 setData
;若是是双向绑定中的父组件数据,则能够触发一个事件去通知父组件去更新对应的值。app
因此根据上面的描述,父组件须要有个监听函数,子组件须要有个智能的 setData
函数。不防将父组件的监听函数 命名为 onSyncAttrUpdate
,将子组件的智能 setData
函数命名为 setDataSmart
,则能够有以下代码:框架
// 父组件
Component({
methods: {
onSyncAttrUpdate(e) {
this.setData(e.detail) // 子组件传来的须要更新的数据
}
}
})
复制代码
<!-- 父组件的模板 -->
<child childAttr="{{parentAttr}}" syncAttrMap="childAttr=parentAttr" bind:syncAttrUpdate="onSyncAttrUpdate" />
复制代码
// 子组件
Component({
properties: {
childAttr: String,
syncAttrMap: String
},
methods: {
// 子组件更新数据时,只要调用此方法便可,而不是 `setData`
setDataSmart(data) {
// splitDataBySyncAttrMap 函数的实现过程就不说了,只是将对象拆分,你们应该都能实现
let {parentData, innerData} = splitDataBySyncAttrMap(data, this.data.syncAttrMap)
// 内部数据使用 setData 更新
if (Object.keys(innerData).length) {
this.setData(innerData) // setData 中还支持 callback 的回调,为了简化代码,这里不讨论
}
// 双向绑定的父组件数据触发事件让父组件本身去更新
if (Object.keys(parentData).length) {
this.triggerEvent('syncAttrUpdate', parentData)
}
}
}
})
复制代码
到此,一个简单的双向绑定功能就完成了。可是因为子组件也有可能包含其它组件,也就是说子组件也能够是父组件,而父组件一样也 能够是子组件。因此上面的 onSyncAttrUpdate
setDataSmart
函数须要在每一个组件中都实现,因此不防 定义一个公共对象 BaseComponent
来实现上面的全部功能,如:
// BaseComponent
const BaseComponent = {
properties: {
syncAttrMap: String
},
methods: {
setDataSmart() {
// ...
},
onSyncAttrUpdate() {
// ...
}
}
}
复制代码
而后将 BaseComponent minin 到每一个组件的对象上去就能够了;另外小程序中还有一个特殊的组件:Page,虽然 Page 和 Component 结构是两样的, 但它也应该算是一个组件,不过它必定是父组件,不多是别的组件的子组件,因此还须要将 onSyncAttrUpdate
方法写了全部的 Page 定义中。 全部这些就是 minapp 的双向绑定的基本原理了。
等等,最后还有一件事:wxml 模板,不能让用户每次写双向绑定的时候都要写那么复杂语句吧?固然不用,minapp 在编译时,会将模板作个简单的转化:
<child childAttr.sync="parentAttr" />
<!-- 因为属性名 syncAttrMap 是固定的,因此彻底能够经过编译手段,将上面的模板转成下面这个模板 -->
<child childAttr="{{parentAttr}}" syncAttrMap="childAttr=parentAttr" />
复制代码
谢谢,文章到此结束,欢迎关注 minapp:从新定义微信小程序的开发
2019年,iKcamp原创新书《Koa与Node.js开发实战》已在京东、天猫、亚马逊、当当开售啦!