📝尤雨溪对逻辑复用的讲解 - 从 Mixin、高阶组件到 Hooks

本文记录《B站·跟尤雨溪一起解读Vue3源码》中,尤雨溪对于逻辑复用演化过程的讲解

假设我们有这么一个组件,[x, y]这一坐标会根据鼠标移动而变化,x,y是响应式的。 我们想把这段逻辑抽取出来,让它可以复用,有什么方法?

js 复制代码
const App = {
  template: `{{ x }} {{ y }}`,
  data() {
    return { x: 0, y: 0 }
  },
  methods: {
    update(e) {
      this.x = e.pageX
      this.y = e.pageY
    }
  },
  mouted() {
    window.addEventListener('mousemove', this.update)
  },
  unmounted() {
    window.removeEventListener('mousemove', this.update)
  }
}

1. mixin

第一种方法,使用mixin。提取mixin很容易,但是如果你有很多个mixin,会出现以下问题:

  • 注入源不明:不知道属性、方法来自哪个mixin
  • 命名冲突
js 复制代码
const MouseMixin = {
  data() {
    return { x: 0, y: 0 }
  },
  methods: {
    update(e) {
      this.x = e.pageX
      this.y = e.pageY
    }
  },
  mouted() {
    window.addEventListener('mousemove', this.update)
  },
  unmounted() {
    window.removeEventListener('mousemove', this.update)
  }
}

const App = {
  mixins: [MouseMixin, mixinA, mixinB, mixnC], // 存在多个mixin时,会很混乱
  template: `{{ x }} {{ y }}`,
}

2. 高阶组件

因为mixin的缺点很明显,React在很早的时候就移除了Mixin。 但是面对逻辑复用,又没有很好的解决方法,于是提出来了高阶组件。 高阶组件的用法:高阶组件包含了需用逻辑复用的变量和方法,并包裹住 Inner 组件,同时还要把x,y传递给 Inner 组件。

js 复制代码
function withMouse(Inner) {
  data() {
    return { x: 0, y: 0 }
  },
  methods: {
    update(e) {
      this.x = e.pageX
      this.y = e.pageY
    }
  },
  mouted() {
    window.addEventListener('mousemove', this.update)
  },
  unmounted() {
    window.removeEventListener('mousemove', this.update)
  },
  render() {
    return h(Inner, {
      x: this.x, 
      y: this.y
    })
  }
}

const App = withMouse({
  props: ['x', 'y'],
  template: `{{ x }} {{ y }}`,
})

相比于mixin,高阶组件有了自己的命名空间,不用担心命名冲突的问题。x, y的来源也指向了withMouse 但依然不能改变所有问题,假如有多个高阶组件包裹。我们依然会得到很多props,同样也无法知道哪个props来自哪个高阶组件

js 复制代码
const App = withFoo(withAnother(withMouse({
  props: ["x", "y", "foo", "bar"]
})))

3. slot

在vue中,跟高阶组件相似的是slot

js 复制代码
function withMouse(Inner) {
  data() {
    return { x: 0, y: 0 }
  },
  methods: {
    update(e) {
      this.x = e.pageX
      this.y = e.pageY
    }
  },
  mouted() {
    window.addEventListener('mousemove', this.update)
  },
  unmounted() {
    window.removeEventListener('mousemove', this.update)
  },
  // template: `<slot :x="x" :y="y" />`,
  render() {
    return this.$slots.default && this.$slots.default({
      x: this.x,
      y: this.y
    })
  }
}

const App = {
  template: `<Mouse v-slot="{x, y}">{{ x }} {{ y }}</Mouse>`
}

存在多个slot时会是,比高阶函数要好的是,我们知道x,y的来源是<Mouse />。但slot额外产生了组件,增加了性能开销

js 复制代码
const App = {
  component: { Mouse, Foo },
  template: `
    <Mouse v-slot="{ x, y }">
      <Foo v-slot="{ foo }">
        {{ x }} {{ y }} {{ foo }}
      </Foo>
    </Mouse>
  `
}

4. Hooks

React Class Componentvue2 option,把data, method都绑定在一起,把它们挪到组件外是一个代价很大的行为。 意识到高阶组件也不是什么灵丹妙药后,React 提出用 Hooks 彻底取代 Class Component,开启了组件逻辑表达和逻辑复用的新范式,完美解决了上面的问题

js 复制代码
function useFeatureA() {
  const foo = ref(0)
  const plusone = computed(() => foo.value + 1)
  return { foo, plusone }
}

function useFeatureB() {
  const bar = ref(1)
  return { bar }
}

export default {
  template: `{{ event.count }}`,
  props: ["id"],
  setup(props) {
    const { foo, plusone } = useFeatureA()
    const { bar } = useFeatureB()
  }
}

参考资料

相关推荐
Python私教11 分钟前
Pure-Admin-Thin 深度解析:完整版和精简版到底怎么选?
vue.js·人工智能·开源
ayqy贾杰3 小时前
Cursor SDK发布!开发者可直接搬走其内核
前端·vue.js·面试
李白的天不白3 小时前
vue 数据格式问题
前端·vue.js·windows
小白蒋博客3 小时前
【ai开发段永平投资理财的知识图谱网站】第一天:搭 Vite + Vue 项目,跑通 Hello World
vue.js·人工智能·trae
@yanyu66613 小时前
登录注册功能-明文
vue.js·springboot
滕青山17 小时前
在线PDF拆分工具核心JS实现
前端·javascript·vue.js
光影少年1 天前
前端在页面渲染优化和组件优化经验?
前端·vue.js·react.js·前端框架
李白的天不白1 天前
VUE依赖配置问题
前端·javascript·vue.js
小智社群1 天前
获取贝壳新房列表
前端·javascript·vue.js
一 乐1 天前
茶叶商城|基于springboot + vue茶叶商城系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·茶叶商城系统