Vue3 的函数式编程

Vue3 的函数式编程

Vue3.0 正式版已经于9月底发布,其中,Vue3 新增的composition-api 是我们讨论的大热门,甚至不少 react 开发者都对其赞不绝口。那么,究竟什么是 函数式编程composition-api 又厉害在哪里捏?

注:看本文章之前,你需要了解一下 Vue3 的 composition-api

动机

逻辑的组合复用

组件 API 设计所面对的核心问题之一就是如何组织逻辑,以及如何在多个组件之间抽取和复用逻辑。

在此之前,Vue2.x 的 Mixins 以及 react 的高阶组件等模式都可以实现逻辑的组合与复用,但它们都存在数据来源不清晰命名空间冲突以及性能的问题。写到这里,要向 react 致敬,因为 React Hooks 的出现是革命性的,而 Vue3 的 composition-api 提供的全新的逻辑复用方案也是受到了 React Hooks 的启发。

我们来观察一下尤雨溪给出来的“用组合函数来封装鼠标位置侦听逻辑”例子:

function useMouse() {
  const x = ref(0)
  const y = ref(0)
  const update = e => {
    x.value = e.pageX
    y.value = e.pageY
  }
  onMounted(() => {
    window.addEventListener('mousemove', update)
  })
  onUnmounted(() => {
    window.removeEventListener('mousemove', update)
  })
  return { x, y }
}

// 在组件中使用该函数
const Component = {
  setup() {
    const { x, y } = useMouse()
    // 与其它函数配合使用
    const { z } = useOtherLogic()
    return { x, y, z }
  },
  template: `<div>{{ x }} {{ y }} {{ z }}</div>`
}

从例子中可以看到 composition-api

  • 暴露给模版的属性来源清晰(从函数返回);
  • 返回值可以被任意重命名,所以不存在命名空间冲突;
  • 没有创建额外的组件实例所带来的性能损耗。

演示

让我们结束枯燥的解释,来感受一下 使用函数式编程的Vue3使用模板语法的Vue2 究竟能有哪些区别。我们从一个简单的 todolist 的例子来对比一下

首先我们使用 Vue2 的模板语法来实现:

<template>
  <div class="home">
    <form>
      <input type="text" v-model="stu.id" />
      <input type="text" v-model="stu.name" />
      <input type="text" v-model="stu.age" />
      <button type="submit" @click="handleAdd">添加</button>
    </form>
    <ul>
      <li v-for="(item, index) in students" :key="item.id">
        {{ item.name }},{{ item.age }}岁,
        <button @click="handleDelete(index)">删除</button>
      </li>
    </ul>
  </div>
</template>

<script>
export default {
  data() {
    return {
      students: [
        { id: 1, name: "张三", age: 10 },
        { id: 2, name: "李四", age: 11 },
        { id: 3, name: "王五", age: 12 },
      ],
      stu: {},
    };
  },
  methods: {
    handleDelete(i) {
      this.students = this.students.filter((item, index) => index != i);
    },
    handleAdd(e) {
      e.preventDefault();
      this.students.push(this.stu);
      this.stu = {};
    },
  },
};
</script>

以上代码使用 Vue2 的模板语法实现了 todolist 的简单的新增和删除功能。

现在我们使用 Vue3 重写一下:

<template>
  <form>
    <input type="text" v-model="state.stu.id" />
    <input type="text" v-model="state.stu.name" />
    <input type="text" v-model="state.stu.age" />
    <button type="submit" @click="handleAdd">添加</button>
  </form>
  <ul>
    <li v-for="(item, index) in state.students" :key="item.id">
      {{ item.name }},{{ item.age }}岁,
      <button @click="handleDelete(index)">删除</button>
    </li>
  </ul>
</template>

<script>
import { reactive } from "vue";
export default {
  setup() {
    // 创建响应式初始数据
    let state = reactive({
      students: [
        { id: 1, name: "张三", age: 10 },
        { id: 2, name: "李四", age: 11 },
        { id: 3, name: "王五", age: 12 },
      ],
      stu: {},
    });

    // 删除方法
    function handleDelete(i) {
      state.students = state.students.filter((item, index) => index != i);
    }

    // 添加方法
    function handleAdd(e) {
      e.preventDefault();
      state.students.push(state.stu);
      state.stu = {};
    }

    // 将数据和方法返回,暴露给模板使用
    return {
      state,
      handleDelete,
      handleAdd,
    };
  },
};
</script>

我们来观察以下 Vue3 的代码,发现我们定义的所有数据和逻辑都放在了 setup 函数里 —— 这难道不是使代码的可读性更差了吗?

当然不是,composition-api 又叫 “组合api” ,我们来看一下当把它逻辑组合之后,我们的 js 代码是什么样子:

<script>
import { reactive } from "vue";
export default {
  setup() {
    // 使用抽离后的移除逻辑
    let { state, handleDelete } = useRemoveStudent();
    // 使用抽离后的新增逻辑
    let { handleAdd } = useAddStudent(state);

    // 将数据和方法返回,暴露给模板使用
    return {
      state,
      handleDelete,
      handleAdd,
    };
  },
};
// 移除功能的逻辑-----------------------------
function useRemoveStudent() {
  // 创建响应式初始数据
  let state = reactive({
    students: [
      { id: 1, name: "张三", age: 10 },
      { id: 2, name: "李四", age: 11 },
      { id: 3, name: "王五", age: 12 },
    ],
    stu: {},
  });

  // 删除方法
  function handleDelete(i) {
    state.students = state.students.filter((item, index) => index != i);
  }

  return {
    state,
    handleDelete,
  };
}

// 新增功能的逻辑------------------------------
function useAddStudent(state) {
  // 添加方法
  function handleAdd(e) {
    e.preventDefault();
    state.students.push(state.stu);
    state.stu = {};
  }
  return {
    handleAdd,
  };
}
</script>

我们主要观察一下 setup 函数,会发现它居然是如此的优雅!所有单独的功能模块全部都独立开来,最后在 setup 函数中统一暴露给模板使用。我们甚至可以把组合后的逻辑放在单独的js文件中,这样会使我们的代码逻辑更直观,可维护性更高,复用性更强,就像我们文章开头所说的那样。

让我们再直观地看一眼 composition-api 的好处:

composition-api

太优雅了。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,468评论 5 473
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,620评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,427评论 0 334
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,160评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,197评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,334评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,775评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,444评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,628评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,459评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,508评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,210评论 3 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,767评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,850评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,076评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,627评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,196评论 2 341