Vue 3 的<script setup> 和 Vue 2 的 Options API的关系

1. 设计理念的差异

  • Vue 2 的 Options API

    基于"选项式"设计,将组件逻辑按照功能(如 datamethodscomputedwatch、生命周期钩子等)拆分为不同的选项对象,所有逻辑通过 this 关联到组件实例。

    优点是结构清晰,适合简单组件;但复杂组件中,同一业务逻辑可能分散在多个选项中(如一个表单验证逻辑可能涉及 datamethodswatch),维护成本较高。

  • <script setup>

    基于 Vue 3 推出的组合式 API(Composition API) ,通过"逻辑组合"的方式组织代码,允许将相关的逻辑(如数据、方法、副作用)聚合在一起,而非按选项拆分。

    简化了组合式 API 的写法(无需手动返回变量、自动暴露给模板等)

2、实例对比

以下通过"计数器"和"用户信息展示"两个例子,直观对比两者的写法差异。

例 1:基础计数器(简单逻辑)

Vue 2 Options API 写法

vue 复制代码
<template>
  <div>
    <p>计数:{{ count }}</p>
    <button @click="increment">+1</button>
  </div>
</template>

<script>
export default {
  // 响应式数据
  data() {
    return {
      count: 0
    }
  },
  // 方法
  methods: {
    increment() {
      this.count++ // 通过 this 访问数据
    }
  },
  // 生命周期
  mounted() {
    console.log('组件挂载完成,初始计数:', this.count)
  }
}
</script>

Vue 3 <script setup> 写法

vue 复制代码
<template>
  <div>
    <p>计数:{{ count }}</p>
    <button @click="increment">+1</button>
  </div>
</template>

<script setup>
// 导入响应式 API
import { ref, onMounted } from 'vue'

// 响应式数据(替代 data)
const count = ref(0)

// 方法(直接定义,无需包裹在 methods 中)
const increment = () => {
  count.value++ // 通过 .value 访问 ref 数据
}

// 生命周期(替代 mounted 选项)
onMounted(() => {
  console.log('组件挂载完成,初始计数:', count.value)
})
</script>

核心差异

  • Options API 中数据和方法需放在固定选项中,通过 this 关联;
  • <script setup> 中直接定义变量和函数,通过 ref 等 API 创建响应式数据,生命周期通过导入的钩子函数实现。
例 2:用户信息展示(含计算属性和 props)

Vue 2 Options API 写法

vue 复制代码
<template>
  <div>
    <p>用户名:{{ fullName }}</p>
    <p>年龄:{{ age >= 18 ? '成年' : '未成年' }}</p>
  </div>
</template>

<script>
export default {
  // 接收父组件传参
  props: {
    firstName: String,
    lastName: String,
    age: Number
  },
  // 计算属性
  computed: {
    fullName() {
      return `${this.firstName} ${this.lastName}` // 依赖 props
    }
  },
  // 监听数据变化
  watch: {
    age(newVal) {
      console.log('年龄变化:', newVal)
    }
  }
}
</script>

Vue 3 <script setup> 写法

vue 复制代码
<template>
  <div>
    <p>用户名:{{ fullName }}</p>
    <p>年龄:{{ age >= 18 ? '成年' : '未成年' }}</p>
  </div>
</template>

<script setup>
import { computed, watch } from 'vue'

// 接收父组件传参(替代 props 选项)
const props = defineProps({
  firstName: String,
  lastName: String,
  age: Number
})

// 计算属性(替代 computed 选项)
const fullName = computed(() => {
  return `${props.firstName} ${props.lastName}` // 直接访问 props
})

// 监听(替代 watch 选项)
watch(
  () => props.age,
  (newVal) => {
    console.log('年龄变化:', newVal)
  }
)
</script>
相关推荐
万少2 小时前
HarmonyOS 开发必会 5 种 Builder 详解
前端·harmonyos
橙序员小站4 小时前
Agent Skill 是什么?一文讲透 Agent Skill 的设计与实现
前端·后端
炫饭第一名7 小时前
速通Canvas指北🦮——基础入门篇
前端·javascript·程序员
王晓枫7 小时前
flutter接入三方库运行报错:Error running pod install
前端·flutter
符方昊7 小时前
React 19 对比 React 16 新特性解析
前端·react.js
ssshooter7 小时前
又被 Safari 差异坑了:textContent 拿到的值居然没换行?
前端
曲折7 小时前
Cesium-气象要素PNG色斑图叠加
前端·cesium
Forever7_7 小时前
Electron 淘汰!新的桌面端框架 更强大、更轻量化
前端·vue.js
不会敲代码17 小时前
前端组件化样式隔离实战:React CSS Modules、styled-components 与 Vue scoped 对比
css·vue.js·react.js
Angelial7 小时前
Vue3 嵌套路由 KeepAlive:动态缓存与反向配置方案
前端·vue.js