vue.js

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript类库 > vue.js > Vue3 Composition API

Vue3中Composition API的原理与实战指南

作者:反正我还没长大

Composition API为开发者提供了一种全新的组织组件逻辑的方式,本文将深入探讨Vue3中Composition API的实际应用,帮助开发者掌握这一强大工具

还记得第一次接触Composition API时的困惑吗?"这不就是把data、methods写在一个setup函数里?"两年的深度实战后,我发现这种想法大错特错。Composition API带来的不仅是语法变化,更是前端开发范式的根本革新。今天,我想通过真实项目案例,带你领悟Composition API的精髓。

开篇反思:为什么需要Composition API?

从一个血泪教训说起

去年我接手了一个遗留的Vue2项目,单个组件文件竟然有1500多行代码!让我们看看这个"怪物"组件的结构:

// 某个用户管理页面 - 典型的Options API意大利面条代码
export default {
  data() {
    return {
      // 用户数据 (50行)
      userList: [],
      currentUser: {},
      userForm: { /* 巨大的表单对象 */ },
      
      // 分页数据 (20行)
      pagination: { /* ... */ },
      
      // 搜索数据 (30行)
      searchFilters: { /* ... */ },
      
      // 权限数据 (40行)
      permissions: { /* ... */ },
      
      // 上传数据 (25行)
      uploadConfig: { /* ... */ },
      
      // 还有更多... (200+行data)
    }
  },
  
  computed: {
    // 300+行的computed,各种逻辑混杂
    filteredUsers() { /* 复杂的过滤逻辑 */ },
    userStats() { /* 统计逻辑 */ },
    permissionMatrix() { /* 权限计算 */ },
    // ... 更多computed
  },
  
  methods: {
    // 800+行的methods,什么都有
    fetchUsers() { /* 获取用户 */ },
    validateForm() { /* 表单验证 */ },
    uploadFile() { /* 文件上传 */ },
    calculatePermissions() { /* 权限计算 */ },
    // ... 几十个方法
  }
}

维护这样的代码简直是噩梦:

Composition API的解决思路

Composition API的核心思想是按功能逻辑组织代码,而不是按选项类型组织:

// 重构后的清爽代码
<script setup>
// 每个功能都是独立、可复用的逻辑单元
import { useUserManagement } from './composables/useUserManagement'
import { usePagination } from './composables/usePagination' 
import { useSearch } from './composables/useSearch'
import { usePermissions } from './composables/usePermissions'
import { useFileUpload } from './composables/useFileUpload'

// 声明式地组合功能
const users = useUserManagement()
const pagination = usePagination({ pageSize: 20 })
const search = useSearch(['name', 'email', 'role'])
const permissions = usePermissions()
const upload = useFileUpload({ accept: '.jpg,.png' })

// 功能间的协调逻辑也变得清晰
const handleSearch = async () => {
  pagination.reset()
  await users.fetch({
    ...search.params,
    page: pagination.current,
    size: pagination.size
  })
}
</script>

立即显现的优势:

一、深入理解:setup函数的运行机制

setup到底在做什么

很多人以为setup只是把data、methods包装了一下,这种理解太浅了。让我用一个例子来说明setup的深层含义:

// 传统思维:这只是语法糖?
<script setup>
const count = ref(0)
const increment = () => count.value++
</script>

// 实际上编译后的代码
export default {
  setup() {
    const count = ref(0)
    const increment = () => count.value++
    
    // 返回的对象会被暴露给模板
    return {
      count,
      increment
    }
  }
}

setup函数的特殊之处:

1. 执行时机的精妙设计

// setup的执行时机
export default {
  beforeCreate() {
    console.log('1. beforeCreate')
  },
  setup() {
    console.log('2. setup执行') // 在beforeCreate和created之间
    
    onBeforeMount(() => {
      console.log('4. beforeMount')
    })
    
    onMounted(() => {
      console.log('5. mounted')
    })
  },
  created() {
    console.log('3. created')
  }
}

为什么要在这个时机执行?

因为setup需要在响应式系统初始化之后,但在实例创建之前运行。这样既能使用响应式API,又能影响实例的创建过程。

2. 作用域的独立性

// 每个组件实例都有独立的setup作用域
<script setup>
// 这个变量只属于当前组件实例
let privateCounter = 0

const publicCounter = ref(0)

// 这个函数也是私有的,外部无法访问
function internalHelper() {
  privateCounter++
}

// 只有通过defineExpose才能暴露给外部
defineExpose({
  publicCounter,
  reset: () => {
    publicCounter.value = 0
    privateCounter = 0
  }
})
</script>

从心智模型理解Composition API

传统Options API的心智模型

// Options API:按选项类型分组
const component = {
  data: {
    // 所有数据放这里
    userInfo: {},
    products: [],
    cart: {}
  },
  computed: {
    // 所有计算属性放这里
    userName() { return this.userInfo.name },
    productCount() { return this.products.length },
    cartTotal() { return this.cart.total }
  },
  methods: {
    // 所有方法放这里
    fetchUser() { /* */ },
    fetchProducts() { /* */ },
    addToCart() { /* */ }
  }
}

这种组织方式类似于按文件类型整理的文件夹:

项目文件夹/
├── 所有图片/
├── 所有文档/
├── 所有视频/
└── 所有代码/

当你要找"用户管理相关的内容"时,需要在多个文件夹中翻找。

Composition API的心智模型

// Composition API:按功能逻辑分组
function useUserManagement() {
  const userInfo = ref({})
  const userName = computed(() => userInfo.value.name)
  const fetchUser = async () => { /* */ }
  
  return { userInfo, userName, fetchUser }
}

function useProductManagement() {
  const products = ref([])
  const productCount = computed(() => products.value.length)
  const fetchProducts = async () => { /* */ }
  
  return { products, productCount, fetchProducts }
}

function useCartManagement() {
  const cart = ref({})
  const cartTotal = computed(() => cart.value.total)
  const addToCart = (product) => { /* */ }
  
  return { cart, cartTotal, addToCart }
}

这种组织方式类似于按项目功能整理的文件夹:

项目文件夹/
├── 用户管理/
│   ├── 用户信息.jpg
│   ├── 用户文档.doc
│   └── 用户代码.js
├── 产品管理/
└── 购物车管理/

想找什么功能,直接去对应的文件夹即可。

深入剖析:Composition的三个层次

在实际项目中,我总结出Composition API有三个应用层次:

层次1:基础组合(替代data、methods)

<script setup>
// 最简单的状态管理
const loading = ref(false)
const data = ref([])

const fetchData = async () => {
  loading.value = true
  try {
    data.value = await api.getData()
  } finally {
    loading.value = false
  }
}

onMounted(fetchData)
</script>

这个层次只是语法的改变,还没有体现出Composition API的真正优势。

层次2:逻辑提取(创建可复用的函数)

// composables/useAsyncData.js
export function useAsyncData(apiFn) {
  const loading = ref(false)
  const data = ref(null)
  const error = ref(null)
  
  const execute = async (...args) => {
    loading.value = true
    error.value = null
    try {
      data.value = await apiFn(...args)
    } catch (err) {
      error.value = err
    } finally {
      loading.value = false
    }
  }
  
  return { loading, data, error, execute }
}

// 在组件中使用
<script setup>
import { useAsyncData } from '@/composables/useAsyncData'
import { api } from '@/api'

const { loading, data, error, execute: fetchUsers } = useAsyncData(api.getUsers)
const { loading: productLoading, data: products, execute: fetchProducts } = useAsyncData(api.getProducts)

onMounted(() => {
  fetchUsers()
  fetchProducts()
})
</script>

这个层次开始体现复用性,但还是比较简单的逻辑。

层次3:复杂状态编排(企业级应用)

// composables/useUserManagement.js
export function useUserManagement() {
  // 状态管理
  const users = ref([])
  const currentUser = ref(null)
  const loading = ref(false)
  
  // 缓存策略
  const cache = new Map()
  const cacheKey = computed(() => {
    return `users_${JSON.stringify(searchParams.value)}`
  })
  
  // 搜索参数
  const searchParams = ref({
    keyword: '',
    role: '',
    status: 'active'
  })
  
  // 依赖注入
  const authStore = inject('authStore')
  const notificationBus = inject('notificationBus')
  
  // 复杂的计算逻辑
  const filteredUsers = computed(() => {
    return users.value.filter(user => {
      if (!authStore.hasPermission('view_user', user)) return false
      if (searchParams.value.keyword && !user.name.includes(searchParams.value.keyword)) return false
      if (searchParams.value.role && user.role !== searchParams.value.role) return false
      return true
    })
  })
  
  // 副作用管理
  watch(searchParams, async (newParams, oldParams) => {
    // 防抖搜索
    await debounce(() => fetchUsers(), 300)
  }, { deep: true })
  
  // 缓存的获取逻辑
  const fetchUsers = async () => {
    const key = cacheKey.value
    
    if (cache.has(key)) {
      users.value = cache.get(key)
      return
    }
    
    loading.value = true
    try {
      const data = await userApi.getUsers(searchParams.value)
      users.value = data
      cache.set(key, data)
      
      notificationBus.emit('users:fetched', data)
    } catch (error) {
      notificationBus.emit('error', error.message)
    } finally {
      loading.value = false
    }
  }
  
  // 清理逻辑
  onUnmounted(() => {
    cache.clear()
  })
  
  return {
    // 状态
    users: readonly(users),
    currentUser: readonly(currentUser),
    loading: readonly(loading),
    searchParams,
    
    // 计算属性
    filteredUsers,
    
    // 方法
    fetchUsers,
    setCurrentUser: (user) => currentUser.value = user,
    clearSearch: () => {
      searchParams.value = { keyword: '', role: '', status: 'active' }
    }
  }
}

这个层次才是Composition API的真正威力所在:

二、构建企业级的Composition Hook

2.1 用户管理Hook的完整实现

// hooks/useUser.js
import { ref, computed, watch, provide, inject } from 'vue'
import { userApi } from '@/api/user'
import { useLocalStorage } from '@/hooks/useLocalStorage'
import { useEventBus } from '@/hooks/useEventBus'

// 全局用户状态管理
const USER_INJECTION_KEY = Symbol('user')

export function createUserProvider() {
  const user = ref(null)
  const loading = ref(false)
  const error = ref(null)
  
  // 持久化用户token
  const { value: token, setValue: setToken, removeValue: removeToken } = 
    useLocalStorage('user_token', '')
  
  // 事件总线
  const { emit, on } = useEventBus()
  
  // 计算属性
  const isAuthenticated = computed(() => !!user.value?.id)
  const userName = computed(() => user.value?.name || '游客')
  const userAvatar = computed(() => user.value?.avatar || '/default-avatar.png')
  const permissions = computed(() => user.value?.permissions || [])
  
  // 检查权限
  const hasPermission = (permission) => {
    return permissions.value.includes(permission) || permissions.value.includes('admin')
  }
  
  // 登录方法
  const login = async (credentials) => {
    try {
      loading.value = true
      error.value = null
      
      const response = await userApi.login(credentials)
      
      user.value = response.user
      setToken(response.token)
      
      // 发布登录成功事件
      emit('user:login', user.value)
      
      return response
    } catch (err) {
      error.value = err.message
      throw err
    } finally {
      loading.value = false
    }
  }
  
  // 登出方法
  const logout = async () => {
    try {
      await userApi.logout()
    } catch (err) {
      console.warn('Logout API failed:', err)
    } finally {
      user.value = null
      removeToken()
      emit('user:logout')
    }
  }
  
  // 获取用户信息
  const fetchUserInfo = async () => {
    if (!token.value) return
    
    try {
      loading.value = true
      const userInfo = await userApi.getUserInfo()
      user.value = userInfo
    } catch (err) {
      // token可能已过期
      if (err.status === 401) {
        logout()
      }
      error.value = err.message
    } finally {
      loading.value = false
    }
  }
  
  // 更新用户信息
  const updateProfile = async (profileData) => {
    try {
      loading.value = true
      const updatedUser = await userApi.updateProfile(profileData)
      user.value = { ...user.value, ...updatedUser }
      emit('user:profile-updated', user.value)
      return updatedUser
    } catch (err) {
      error.value = err.message
      throw err
    } finally {
      loading.value = false
    }
  }
  
  // 监听token变化,自动获取用户信息
  watch(token, (newToken) => {
    if (newToken) {
      fetchUserInfo()
    } else {
      user.value = null
    }
  }, { immediate: true })
  
  const userProvider = {
    user: readonly(user),
    loading: readonly(loading),
    error: readonly(error),
    isAuthenticated,
    userName,
    userAvatar,
    permissions,
    hasPermission,
    login,
    logout,
    fetchUserInfo,
    updateProfile
  }
  
  provide(USER_INJECTION_KEY, userProvider)
  
  return userProvider
}

// 在组件中使用
export function useUser() {
  const userProvider = inject(USER_INJECTION_KEY)
  if (!userProvider) {
    throw new Error('useUser must be used within a user provider')
  }
  return userProvider
}

2.2 异步数据获取Hook

// hooks/useAsyncData.js
import { ref, computed, watch, unref } from 'vue'

export function useAsyncData(fetcher, options = {}) {
  const {
    immediate = true,
    resetOnExecute = true,
    shallow = true,
    throwError = false
  } = options
  
  const data = shallow ? shallowRef(null) : ref(null)
  const loading = ref(false)
  const error = ref(null)
  const executeCount = ref(0)
  
  const execute = async (...args) => {
    try {
      executeCount.value++
      loading.value = true
      
      if (resetOnExecute) {
        error.value = null
      }
      
      const result = await fetcher(...args)
      data.value = result
      
      return result
    } catch (err) {
      error.value = err
      if (throwError) {
        throw err
      }
      return null
    } finally {
      loading.value = false
    }
  }
  
  // 重试逻辑
  const retry = () => execute()
  
  // 刷新数据
  const refresh = () => execute()
  
  // 清空数据
  const clear = () => {
    data.value = null
    error.value = null
  }
  
  // 计算状态
  const isFirstLoad = computed(() => executeCount.value === 0)
  const hasData = computed(() => data.value != null)
  const hasError = computed(() => error.value != null)
  
  if (immediate) {
    execute()
  }
  
  return {
    data: readonly(data),
    loading: readonly(loading),
    error: readonly(error),
    executeCount: readonly(executeCount),
    isFirstLoad,
    hasData,
    hasError,
    execute,
    retry,
    refresh,
    clear
  }
}

// 使用示例
export function useProducts() {
  const { 
    data: products, 
    loading, 
    error, 
    execute: fetchProducts,
    refresh: refreshProducts 
  } = useAsyncData(() => productApi.getList())
  
  const productCount = computed(() => products.value?.length || 0)
  
  const searchProducts = async (keyword) => {
    return fetchProducts({ keyword })
  }
  
  return {
    products,
    loading,
    error,
    productCount,
    fetchProducts,
    refreshProducts,
    searchProducts
  }
}

2.3 表单处理Hook

// hooks/useForm.js
import { ref, reactive, computed, watch, nextTick } from 'vue'

export function useForm(initialValues = {}, options = {}) {
  const { validateOnChange = false, validateOnBlur = true } = options
  
  // 表单数据
  const formData = reactive({ ...initialValues })
  
  // 表单验证状态
  const errors = ref({})
  const touched = ref({})
  const validating = ref(false)
  
  // 验证规则
  const rules = ref({})
  
  // 设置验证规则
  const setRules = (newRules) => {
    rules.value = newRules
  }
  
  // 单个字段验证
  const validateField = async (field) => {
    const rule = rules.value[field]
    if (!rule) return true
    
    try {
      validating.value = true
      
      // 支持函数和数组两种规则格式
      const fieldRules = Array.isArray(rule) ? rule : [rule]
      
      for (const fieldRule of fieldRules) {
        if (typeof fieldRule === 'function') {
          const result = await fieldRule(formData[field], formData)
          if (result !== true) {
            errors.value[field] = result
            return false
          }
        } else if (fieldRule.validator) {
          const result = await fieldRule.validator(formData[field], formData)
          if (!result) {
            errors.value[field] = fieldRule.message || '验证失败'
            return false
          }
        }
      }
      
      // 验证通过,清除错误
      delete errors.value[field]
      return true
    } catch (err) {
      errors.value[field] = err.message || '验证出错'
      return false
    } finally {
      validating.value = false
    }
  }
  
  // 全表单验证
  const validate = async () => {
    const fieldNames = Object.keys(rules.value)
    const results = await Promise.all(
      fieldNames.map(field => validateField(field))
    )
    
    return results.every(result => result)
  }
  
  // 设置字段值
  const setFieldValue = (field, value) => {
    formData[field] = value
    
    if (validateOnChange) {
      nextTick(() => validateField(field))
    }
  }
  
  // 设置字段触摸状态
  const setFieldTouched = (field, isTouched = true) => {
    touched.value[field] = isTouched
    
    if (isTouched && validateOnBlur) {
      validateField(field)
    }
  }
  
  // 重置表单
  const resetForm = () => {
    Object.keys(formData).forEach(key => {
      formData[key] = initialValues[key]
    })
    errors.value = {}
    touched.value = {}
  }
  
  // 提交表单
  const submitForm = async (onSubmit) => {
    // 标记所有字段为已触摸
    Object.keys(rules.value).forEach(field => {
      touched.value[field] = true
    })
    
    const isValid = await validate()
    
    if (isValid && onSubmit) {
      return onSubmit(formData)
    }
    
    return isValid
  }
  
  // 计算属性
  const hasErrors = computed(() => Object.keys(errors.value).length > 0)
  const isSubmittable = computed(() => !hasErrors.value && !validating.value)
  
  // 获取字段错误
  const getFieldError = (field) => {
    return touched.value[field] ? errors.value[field] : null
  }
  
  // 监听表单数据变化
  watch(
    () => formData,
    () => {
      if (validateOnChange) {
        Object.keys(touched.value).forEach(field => {
          if (touched.value[field]) {
            validateField(field)
          }
        })
      }
    },
    { deep: true }
  )
  
  return {
    formData,
    errors: readonly(errors),
    touched: readonly(touched),
    validating: readonly(validating),
    hasErrors,
    isSubmittable,
    setRules,
    validateField,
    validate,
    setFieldValue,
    setFieldTouched,
    resetForm,
    submitForm,
    getFieldError
  }
}

三、高级Hook模式与最佳实践

3.1 组合式Hook的设计模式

// hooks/useUserProfile.js - 复合型Hook
import { computed } from 'vue'
import { useUser } from './useUser'
import { useForm } from './useForm'
import { useAsyncData } from './useAsyncData'

export function useUserProfile() {
  // 组合多个基础Hook
  const { user, updateProfile } = useUser()
  
  // 表单处理
  const { 
    formData, 
    errors, 
    setRules, 
    submitForm, 
    resetForm,
    getFieldError 
  } = useForm({
    name: '',
    email: '',
    bio: '',
    avatar: ''
  })
  
  // 设置验证规则
  setRules({
    name: [
      (value) => value ? true : '姓名不能为空',
      (value) => value.length >= 2 ? true : '姓名至少2个字符'
    ],
    email: [
      (value) => value ? true : '邮箱不能为空',
      (value) => /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(value) ? true : '邮箱格式不正确'
    ]
  })
  
  // 头像上传
  const { 
    data: uploadResult, 
    loading: uploading, 
    execute: uploadAvatar 
  } = useAsyncData(async (file) => {
    const formData = new FormData()
    formData.append('avatar', file)
    return uploadApi.uploadAvatar(formData)
  }, { immediate: false })
  
  // 监听用户数据变化,同步到表单
  watch(user, (newUser) => {
    if (newUser) {
      Object.assign(formData, {
        name: newUser.name || '',
        email: newUser.email || '',
        bio: newUser.bio || '',
        avatar: newUser.avatar || ''
      })
    }
  }, { immediate: true })
  
  // 监听头像上传结果
  watch(uploadResult, (result) => {
    if (result?.url) {
      formData.avatar = result.url
    }
  })
  
  // 提交表单
  const handleSubmit = () => {
    return submitForm(async (data) => {
      await updateProfile(data)
      // 可以添加成功提示等逻辑
    })
  }
  
  // 计算属性
  const hasChanges = computed(() => {
    if (!user.value) return false
    
    return Object.keys(formData).some(key => 
      formData[key] !== user.value[key]
    )
  })
  
  return {
    // 表单状态
    formData,
    errors,
    getFieldError,
    hasChanges,
    
    // 头像上传
    uploading,
    uploadAvatar,
    
    // 操作方法
    handleSubmit,
    resetForm
  }
}

3.2 响应式缓存Hook

// hooks/useCache.js
import { ref, computed, watch } from 'vue'

export function useCache(key, fetcher, options = {}) {
  const {
    ttl = 5 * 60 * 1000, // 5分钟缓存
    staleWhileRevalidate = true,
    maxRetries = 3
  } = options
  
  const cache = new Map()
  const loading = ref(false)
  const error = ref(null)
  const retryCount = ref(0)
  
  const getCacheKey = (params) => {
    return typeof key === 'function' ? key(params) : key
  }
  
  const isStale = (cacheEntry) => {
    return Date.now() - cacheEntry.timestamp > ttl
  }
  
  const fetchData = async (params) => {
    const cacheKey = getCacheKey(params)
    const cacheEntry = cache.get(cacheKey)
    
    // 如果有缓存且未过期,直接返回
    if (cacheEntry && !isStale(cacheEntry)) {
      return cacheEntry.data
    }
    
    // 如果启用了staleWhileRevalidate且有过期缓存
    if (staleWhileRevalidate && cacheEntry) {
      // 先返回过期数据,后台重新获取
      setTimeout(() => backgroundFetch(params), 0)
      return cacheEntry.data
    }
    
    return foregroundFetch(params)
  }
  
  const foregroundFetch = async (params) => {
    try {
      loading.value = true
      error.value = null
      
      const data = await fetcher(params)
      const cacheKey = getCacheKey(params)
      
      cache.set(cacheKey, {
        data,
        timestamp: Date.now()
      })
      
      retryCount.value = 0
      return data
    } catch (err) {
      error.value = err
      
      if (retryCount.value < maxRetries) {
        retryCount.value++
        return foregroundFetch(params)
      }
      
      throw err
    } finally {
      loading.value = false
    }
  }
  
  const backgroundFetch = async (params) => {
    try {
      const data = await fetcher(params)
      const cacheKey = getCacheKey(params)
      
      cache.set(cacheKey, {
        data,
        timestamp: Date.now()
      })
    } catch (err) {
      console.warn('Background fetch failed:', err)
    }
  }
  
  const invalidateCache = (params) => {
    const cacheKey = getCacheKey(params)
    cache.delete(cacheKey)
  }
  
  const clearAllCache = () => {
    cache.clear()
  }
  
  return {
    loading: readonly(loading),
    error: readonly(error),
    fetchData,
    invalidateCache,
    clearAllCache
  }
}

// 使用示例
export function useProductList() {
  const { fetchData, loading, error, invalidateCache } = useCache(
    (params) => `products:${JSON.stringify(params)}`,
    (params) => productApi.getList(params),
    { ttl: 10 * 60 * 1000 } // 10分钟缓存
  )
  
  const products = ref([])
  
  const loadProducts = async (params = {}) => {
    try {
      products.value = await fetchData(params)
    } catch (err) {
      console.error('Failed to load products:', err)
    }
  }
  
  const refreshProducts = (params) => {
    invalidateCache(params)
    return loadProducts(params)
  }
  
  return {
    products: readonly(products),
    loading,
    error,
    loadProducts,
    refreshProducts
  }
}

四、性能优化与最佳实践

4.1 避免响应式性能陷阱

// ❌ 错误的做法:过度响应式
export function useBadExample() {
  const heavyData = reactive({
    // 大量复杂嵌套数据
    list: new Array(10000).fill(0).map(i => ({
      id: i,
      details: { /* 复杂对象 */ }
    }))
  })
  
  return { heavyData }
}

// ✅ 正确的做法:按需响应式
export function useGoodExample() {
  // 只对需要响应的部分使用reactive
  const selectedIds = ref(new Set())
  const viewMode = ref('list')
  
  // 大量数据使用shallowRef
  const heavyData = shallowRef([])
  
  // 计算属性只依赖必要的响应式数据
  const selectedItems = computed(() => {
    return heavyData.value.filter(item => 
      selectedIds.value.has(item.id)
    )
  })
  
  const updateData = (newData) => {
    // 手动触发更新
    heavyData.value = newData
  }
  
  return {
    selectedIds,
    viewMode,
    selectedItems,
    updateData
  }
}

4.2 Hook的依赖注入模式

// plugins/composition-providers.js
export function createCompositionProviders(app) {
  // 全局状态提供者
  app.provide('globalStore', createGlobalStore())
  app.provide('userProvider', createUserProvider())
  app.provide('themeProvider', createThemeProvider())
}

// main.js
import { createApp } from 'vue'
import { createCompositionProviders } from './plugins/composition-providers'

const app = createApp(App)
createCompositionProviders(app)
app.mount('#app')

五、总结与最佳实践建议

经过两年多的Composition API实践,我总结了以下最佳实践:

5.1 Hook设计原则

5.2 命名约定

// ✅ 推荐的命名方式
useUser()         // 用户相关逻辑
useAsyncData()    // 异步数据处理
useLocalStorage() // 本地存储
useEventBus()     // 事件总线

// ❌ 避免的命名方式
getUserHook()     // 冗余的Hook后缀
userLogic()       // 不明确的命名
handleUser()      // 与方法命名混淆

5.3 代码组织建议

hooks/
├── core/           # 核心Hook
│   ├── useAsyncData.js
│   ├── useLocalStorage.js
│   └── useEventBus.js
├── business/       # 业务Hook
│   ├── useUser.js
│   ├── useProducts.js
│   └── useOrders.js
├── ui/            # UI相关Hook
│   ├── useModal.js
│   ├── useToast.js
│   └── useForm.js
└── index.js       # 统一导出

Composition API的真正价值在于它让我们能够以一种更加自然、直观的方式组织代码。通过合理的Hook设计,我们可以构建出既高性能又易维护的Vue3应用。

以上就是Vue3中Composition API的原理与实战指南的详细内容,更多关于Vue3 Composition API的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文