我的提示: AIpine 是一个js 库,官网口号是 “一个新的轻量极javascript框架”,其实我之前也没接触过,翻译这篇文章时才注意到

官方地址: [AIpine.js]https://alpinejs.dev


下面开始是译文:

小提示: 在这篇文章中我将使用Vue/AIpine 术语 ,但是我认为此模式可以应用于更多不同的语言框架

前段时间我就碰到了数千行的超大表格。每一行都是单独的 AIpine 组件, 你可以通过点击激活它(css会显示高亮)。如果你点击了另一行,那么前一行激活状态就会处理非激活状态,新行则是激活状态。

问题就是:激活某一个单行居然差不多要耗时整整一秒钟

此类性能问题几乎让这整个效果无法使用,特别是在用键盘操作导航至单元格时。

所以 ,我用一个加载了1万行的页面去测试寻找以找到尽可能的提高性能方法。 不久, 我想出了一个简洁的模式让页面可以立即更新状态。我称它为: Switchboard 模式

下面展示展示…

在此例子中,我不打算用AIpine 。取而代之的是AIpine 底层用到的 vue 提供的一些通用响应式方法。 如果你对 “watchEffect” 和 “ref” 不太熟悉,通过后面的代码片断你应该能凭直觉就知道它们的用法,如果还是不知道,那么 api 文档就在这里查看

假定给我们一个拥有1万行的表格,下面简单的代码会在页面加载时高亮当前激活的 activeRow

  1. let activeRow = 12
  2. document.querySelectorAll('tr').forEach((row) => {
  3. if (row.id === activeRow) {
  4. row.classList.add('active')
  5. } else {
  6. row.classList.remove('active')
  7. }
  8. })

现在,当不同行被点击时,我们可以给行添加点击事件来设置新的高亮行

  1. let activeRow = 12
  2. document.querySelectorAll('tr').forEach((row) => {
  3. row.addEventListener('click', () => {
  4. activeRow = row.id
  5. })
  6. if (row.id === activeRow) {
  7. row.classList.add('active')
  8. } else {
  9. row.classList.remove('active')
  10. }
  11. })

以上代码的问题是,当一个行被点击,当前激活行会更新,但在视觉上我们看不到任何变化。

下面展示了我们可以使用 “reactivity” 让当activeRow 发生变化时,所有行自己触发自身的更新:

  1. import { ref, watchEffect } from 'vue'
  2. let activeRow = ref(12)
  3. document.querySelectorAll('tr').forEach((row) => {
  4. row.addEventListener('click', () => {
  5. activeRow.value = row.id
  6. })
  7. watchEffect(() => {
  8. if (row.id === activeRow.value) {
  9. row.classList.add('active')
  10. } else {
  11. row.classList.remove('active')
  12. }
  13. })
  14. })

上面的代码片断做了这么几件事

  • 用ref 包裹 activeRow 变量,从而使得它可以被响应依赖追踪
  • 循环1万行,添加点击事件,用于改变响应式的 actievRow 变量
  • 注册一个响应式副作用 watchEffect 它会在任意响应依赖变更时重新运行(此处是 activeRow )

这是为何AIpine ( 或Vue 也类似,以我的知识范围内的理解来讲 )能在底层成功工作的原理,如果你要渲染1万行组件,它们全部依赖一个响应式状态比如: “activerRow”

现在,当某个用户点击某行,那么被点击行将变成 active 其它行自动变成 deactivated

问题是: 页面更新超级变

为什么 ?因为每当activeRow变量发生变化时, 1万个watchEffect 回调会被执行。

大多数 app中, 声明一个状态,然后它被子组件,这不成问题。 然而,如果你你正在创建非常多的组件(或“效果”),除了被activeRow状态影响的相关的两行外,其它9998 完全不需要关心状态变更, 这非常低效。

解决方案: 一个响应式的switchboard

响应式 switchboard 这术语是我现在为这个概念创造的。 非常有可能这个模式也许已经有了其它的名称,但是,管它呢…

在当前设置中,我们有单个一个状态,和1万个依赖于此状态的地方。

假如换成一个单独状态,和1万个不同预存的值(和上面一样),我们拥有了1万个不同的状态,每个状态是一个布尔值,代表了每个预设值。举个栗子:

  1. // Before
  2. let activeRow = ref(4)
  3. // After
  4. let rowStates = {
  5. 1: ref(false),
  6. 2: ref(false),
  7. 3: ref(false),
  8. 4: ref(true),
  9. 5: ref(false),
  10. ...
  11. }

让我们稍变动一下上面例子的代码来使用此模式:

  1. import { ref, watchEffect } from 'vue'
  2. let rowStates = {}
  3. document.querySelectorAll('tr').forEach((row) => {
  4. rowStates[row.id] = ref(false)
  5. row.addEventListener('click', () => {
  6. rowStates[row.id].value = true
  7. })
  8. watchEffect(() => {
  9. if (rowStates[row.id].value) {
  10. row.classList.add('active')
  11. } else {
  12. row.classList.remove('active')
  13. }
  14. })
  15. })

好了,现在你能看到, 不同于activeRow存储单一的row ID, 我们使用 rowStates 来存储1万条数据,每条key就是row ID, 每条数据值就是一个响应式的布尔值,代表了当前行是否处于激活状态

这行的通且超级快,现在,由于只点击一行,只有被点击的当前行状态会变更状态(不会影响到其它9999行)

不过还有一个问题 之前 因为 activeRow 只包含引用一个值,相同时间当前只有一个行被允许激活。 前一个行会自动变更为非激活态,因为每行都会自动重新计算。

在这个例子中,“非激活过程”没有触发。 为了让行拥有非激活态,我们需要在rowStates里找到它并标记它的值为false

让我们添加一丢丢代码来实现它:

  1. import { ref, watchEffect } from 'vue'
  2. let rowStates = {}
  3. document.querySelectorAll('tr').forEach((row) => {
  4. rowStates[row.id] = ref(false)
  5. row.addEventListener('click', () => {
  6. // Deactivate the old row...
  7. for (id in rowStates) {
  8. if (rowStates[id].value === true) {
  9. rowStates[id].value = false
  10. return
  11. }
  12. }
  13. rowStates[row.id].value = true
  14. })
  15. watchEffect(() => {
  16. if (rowStates[row.id].value) {
  17. row.classList.add('active')
  18. } else {
  19. row.classList.remove('active')
  20. }
  21. })
  22. })

正如你所看到的,我们添加了一丢丢代码在点击事件内,循环全部的行并设置为非激活态

现在我们加上了非激活态功能 ,但我们的代码依然不高效,每次行被点击,就需要循环rowStates对象的1万项

结果是我们回头在之前优化中使用过的,通过添加一点数据来存储当前激活的行ID。 它类似于基础的暂存,使得我们无需再使用循环了:

  1. import { ref, watchEffect } from 'vue'
  2. let rowStates = {}
  3. let activeRow
  4. document.querySelectorAll('tr').forEach((row) => {
  5. rowStates[row.id] = ref(false)
  6. row.addEventListener('click', () => {
  7. if (activeRow) rowStates[activeRow].value = false
  8. activeRow = row.id
  9. rowStates[row.id].value = true
  10. })
  11. watchEffect(() => {
  12. if (rowStates[row.id].value) {
  13. row.classList.add('active')
  14. } else {
  15. row.classList.remove('active')
  16. }
  17. })
  18. })

得了,现在我们添加 了activeRow 变量,我们搞定了完美高效更新

接近完美,但感觉还差点意思,如果我们能简单抽象一下让我们少做一些跑腿的活。

小的函数 switchboard 它包含了一个值,并返回一些通用函数用于访问和变更这个值

  1. import { watchEffect } from 'vue'
  2. import { switchboard } from 'reactive-switchboard' // Heads up: this isn't on NPM
  3. let { set: activate, is: isActive } = switchboard(12)
  4. document.querySelectorAll('tr').forEach((row) => {
  5. row.addEventListener('click', () => {
  6. activate(row.id)
  7. })
  8. watchEffect(() => {
  9. if (isActive(row.id)) {
  10. row.classList.add('active')
  11. } else {
  12. row.classList.remove('active')
  13. }
  14. })
  15. })

现在通过小小的switchboard 函数, 我们拥有了和之前一样洁净的代码,并且拥有超高效的性能

这里是全部的 switchboard API

  1. import { switchboard } from 'reactive-switchboard' // Heads up: this isn't on NPM
  2. let { get, set, is } = switchboard(12)
  3. // get() returns "12" in this case (non-reactively)
  4. // set(10) sets the internal value to 10
  5. // is(10) runs a reactive comparison to the underlying value

这对于追踪类似于active激活态超级有用,因为只有一个激活状态值了

我也找到了类似的需求,在追踪 ‘selected’ 状态时,这需要多个状态

对于这些需求,我新建了一个通用方法 switchboardSet 它拥有类似 Set 对象的API (可能有更好的名字,但管它呢…)

  1. import { switchboardSet } from 'reactive-switchboard' // Heads up: this isn't on NPM
  2. let { get, add, remove, has, clear } = switchboardSet([12])
  3. // get() returns [12] (non-reactively)
  4. // add(10) sets the internal array to [12, 10]
  5. // remove(10) sets the array back to [12]
  6. // has(12) returns a reactive boolean
  7. // clear() reactively clears the internal array: []

老弟你行了,发现问题,找到解决方法,并抽象它。

我把switchboard源码放到 github上了

自取!

英文原文链接

https://calebporzio.com/reactive-switchboard


转载入注明博客园 王二狗Sheldon

Email: willian12345@126.com

https://github.com/willian12345


响应式的 switchboard:让又大又慢的Vue/AIpine 页面爆快的更多相关文章

  1. 响应式WEB设计的基本原则大总结

    响 应式Web设计对于解决多类型屏幕问题来说是个不错方案,但从印刷的角度来看,其却存在着很多的困难.没有固定的页面尺寸.没有毫米或英寸,没有任何物理 限制,让人感到无从下手.随着建立网站可用的各种小工 ...

  2. 手持式停车收费管理系统全套案例,支持车牌识别,包含了android版app,微信小程序查询,响应式管理后台,云端大数据存储

    先展示几个app效果图片吧,使用起来非常方便,关联了机器的快捷键操作,操作速度提高了不少,摄像头车牌自动识别,车牌识别无网络情况下离线也可以使用   再来一张后台截图,停车场信息完整显示,今日数据实时 ...

  3. 前端必读:Vue响应式系统大PK

    转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 原文参考:https://www.sitepoint.com/vue-3-reactivity-system ...

  4. CSS3与页面布局学习笔记(四)——页面布局大全(负边距、双飞翼、多栏、弹性、流式、瀑布流、响应式布局)

    一.负边距与浮动布局 1.1.负边距 所谓的负边距就是margin取负值的情况,如margin:-100px,margin:-100%.当一个元素与另一个元素margin取负值时将拉近距离.常见的功能 ...

  5. 什么是响应式Web设计?怎样进行?

    http://beforweb.com/node/6/page/0/3 开始第一篇.老规矩,先无聊的谈论天气一类的话题.十一长假,天气也终于开始有些秋天的味道,坐在屋里甚至觉得需要热咖啡.话说两年前也 ...

  6. 响应式Web设计 H5和CSS3实战(第二版)随笔

    第一章 响应式设计基础 1.弹性布局 <meta name = "viewport" content = "width = device-width"&g ...

  7. Web移动前端开发-——bootstarp响应式框架

    移动端WEB开发之响应式布局 1.0 响应式开发原理 1.1 响应式开发原理 就是使用媒体查询针对不同宽度的设备进行布局和样式的设置,从而适配不同设备的目的. 设备的划分情况: 小于768的为超小屏幕 ...

  8. 使用 Responsive Elements 快速构建响应式网站

    Responsive Elements 可以使任何元素来适应和应对他们所占据的区域.这是一个轻量的 JavaScript 库,你可以轻松嵌入到你的项目.元素会更具自己的宽度,自动响应和适应空间的增加或 ...

  9. Bootstrap页面布局5 - 响应式布局(格式)

    旨在优化不同上网设备中页面显示的优化 响应式布局:就是根据浏览窗口的尺寸,改变页面的变化 原理:利用css的media-queries判断浏览窗口的尺寸,在CSS样式表中设置一些规则! 例如: 在&l ...

  10. Vue 响应式总结

    有些时候,不得不想添加.修改数组和对象的值,但是直接添加.修改后getter.setter又失去了. 由于 JavaScript 的限制, Vue 不能检测以下变动的数组: 当你利用索引直接设置一个项 ...

随机推荐

  1. WEB开发日志1

    2020/6/11 23:23 今天做系统时,用到二级菜单,菜单下方放了一个<iframe>标签,但二级菜单的菜单项太多,导致一部分菜单项被<iframe>覆盖,从而无法再选中 ...

  2. 日志注解,基于ruoyi的后置切面改进而来

    有次接口响应时间太长,想知道具体接口执行的时间是多少,于是决定通过注解来实现这个想法,刚好ruoyi本身就提供了完善的日志注解,虽然是采用后置通知,但是完全不影响我们改造它. 想要实现接口耗时的功能, ...

  3. ArcEngine构造多部件

  4. C#开发微信

    C#开发微信门户及应用教程   C#开发微信门户及应用(1)--开始使用微信接口... 6 1.微信账号... 6 2.微信菜单定义... 7 3.接入微信的链接处理... 8 4.使用开发方式创建菜 ...

  5. Spring设计模式——原型模式

    原型模式 原型模式(Prototype Pattern),是指原型实例指定创建对象的种类,并且通过复制这些原型创建新的对象. 原型模式主要适用于以下场景: 类初始化消耗资源较多 使用new生成一个对象 ...

  6. GO语言学习笔记-测试篇 Study for Go ! Chapter ten- Test

    持续更新 Go 语言学习进度中 ...... GO语言学习笔记-类型篇 Study for Go! Chapter one - Type - slowlydance2me - 博客园 (cnblogs ...

  7. D - Swap Free Gym - 102423D 二分图性质:补图最大团 = 点的个数 - 最大匹配数

    题意:给你一个串的某些全排列,没有重的,让你求一个最大的集合能有多少个元素,集合的满足条件:交换一个串的任意两个位置上的字母,不能变成集合里的另一个串. 思路:如果一个串不能通过交换一次字母位置变成另 ...

  8. PicGo + Gitee(码云)实现markdown图床 (转载)

    https://zhuanlan.zhihu.com/p/102594554 备忘录 我配置图床的时候参考的是这篇文章.我暂时使用的是这种方案. 因为考虑到有的文章要多平台发布,我建议你选择markd ...

  9. ThreadLocal部分源码分析和应用场景

    结构演进 早起JDK版本中,ThreadLocal内部结构是一个Map,线程为key,线程在"线程本地变量"中绑定的值为Value.每一个ThreadLocal实例拥有一个Map实 ...

  10. 2020寒假学习笔记15------Spark基础实验

    今天又开始重新做实验六,第一题做的比较顺利,运行结果如下: 等到第二题就出现了各种各样的错误,开始运行telnet localhost 44444命令时出现bash: telnet: command ...