Vue篇.03-组合式API [setup()]

  1. 单文件组件

(1)<script setup>

  • <script setup> 是在单文件组件 (SFC) 中使用组合式 API 的编译时语法糖。当同时使用 SFC 与组合式 API 时该语法是默认推荐

  • 启用该语法,需要在 <script> 代码块上添加 setup attribute,   里面的代码会被编译成组件 setup() 函数的内容

  • 普通的 <script> 只在组件被首次引入时执行一次

  • <script setup> 中的代码会在每次组件实例被创建的时候执行

  • <script setup>需要写在组件中的最前边

(2)入口

  • setup() 钩子是在组件中使用组合式 API 的入口

  • 写法:  setup(){     }

  • 注意:在setup中避免使用this

(3)执行时机

执行时机:组件被创建之前执行

使用setup钩子替代beforeCreate和created函数。所有vue3中没有beforeCreate和created的钩子函数

2.使用

(1)顶层的绑定会被暴露给模板

  • 当使用 <script setup> 时,任何在 <script setup> 声明的顶层的绑定 (包括变量,函数声明,以及 import 导入的内容) 都能在模板中直接使用

  • import 导入的内容也会以同样的方式暴露。可以在模板表达式中直接使用导入的函数,而不需要通过 methods 选项来暴露它

<script setup>
    import ToDoHeader from './SetupToDoHeader.vue'
    // 使用任<script setup> 其中声明的顶层的绑定 (包括变量,函数声明,以及 import 导入的内容) 都能在模板中直接使用, 不需要再使用components, methods, data()等注册:
</script>
<template>
        <ToDoHeader @add="addList" @deleteAll="delAll"> </ToDoHeader>
</template>

(2)props用法

// 用于接收父组件传的的值    props的用法
    // 方式1
    // const props = defineProps(['todoList'])
    // 方式2
    const props = defineProps({
            todoList:{
                type: Array, 
                require: true,
            },
            title:{
                type: String,
                default: '待办事项'
            },
            flag: {
                type: Boolean,
                defalut: false
            }
        })

(3)响应式ref()

响应式 ref():接受一个内部值(number|string),返回一个响应式的、可更改的 ref 对象,此对象只有一个指向其内部值的属性 .value

    //使用前引入
    import { ref,reactive } from 'vue'    
    // 此处声明的是 非响应式的
    const msg = 'Hello!'
    // 使用ref ,使得变量成为 响应式的,  使用前先引入   数组在删除时, 有问题, 故改为对象的形式进行删除
    const todoVal = ref('')
    // 避免使用this   常量修改值时, 需要通过 .value来修改, 在显示时不需要.value

(4)reactive响应式代理

  const obj = reactive({ todoList : [] })
    // reactive(引用类型) 响应式代理
    // const obj = reactive({})

基于响应式对象上的一个属性,创建一个对应的 ref。这样创建的 ref 与其源属性保持同步:改变源属性的值将更新 ref 的值,反之亦然

toRef()

const state = reactive({
  foo: 1,
  bar: 2
})
const fooRef = toRef(state, 'foo')
// 更改该 ref 会更新源属性
fooRef.value++
console.log(state.foo) // 2
// 更改源属性也会更新该 ref
state.foo++
console.log(fooRef.value) // 3

请注意,这不同于:

const fooRef = ref(state.foo)

上面这个 ref 不会和 state.foo 保持同步,因为这个 ref() 接收到的是一个纯数值

toRefs()

将一个响应式对象转换为一个普通对象,这个普通对象的每个属性都是指向源对象相应属性的 ref。每个单独的 ref 都是使用 toRef() 创建的。

const state = reactive({
  foo: 1,
  bar: 2
})

const stateAsRefs = toRefs(state)
/*
stateAsRefs 的类型:{
  foo: Ref<number>,
  bar: Ref<number>
}
*/
// 这个 ref 和源属性已经“链接上了”
state.foo++
console.log(stateAsRefs.foo.value) // 2
stateAsRefs.foo.value++
console.log(state.foo) // 3
//当从组合式函数中返回响应式对象时,toRefs 相当有用。使用它,消费者组件可以解构/展开返回的对象而不会失去响应性:
function useFeatureX() {
  const state = reactive({
    foo: 1,
    bar: 2
  })
  // ...基于状态的操作逻辑
  // 在返回时都转为 ref
  return toRefs(state)
}
// 可以解构而不会失去响应性
const { foo, bar } = useFeatureX()

(5)computed()

//创建一个可写的计算属性 ref:
const count = ref(1)
const plusOne = computed({
  get: () => count.value + 1,
  set: (val) => {
    count.value = val - 1
  }
})

(6)watch()/watchEffect()

watch() 侦听一个或多个响应式数据源,并在数据源变化时调用所给的回调函数

    import { ref , watch, watchEffect} from 'vue'
    const before = ref(0)
    const after = ref(0)
    // watch() 默认是懒侦听的(深度侦听),即仅在侦听源发生变化时才执行回调函数
    watch(props.todoList, (newList, preList)=>{
        after.value = newList.filter(item=>item.done).length
        before.value = newList.length - after.value
    })
    // 立即执行 一个函数,同时响应式地追踪其依赖,并在依赖更改时重新执行
    // watchEffect(()=>console.log(before.value++)) 
    // 停止侦听器
    // const stop = watchEffect(()=>{})
    // stop()

(7)provide和inject

    import { provide } from 'vue'    
    // 向后代传参   provide(键,值)
    // 提供静态值  
    provide('msg','hahahaah')
    // 提供响应式的值
    const count = ref(0)
    provide('count',count)
    // 接收发送方传递的值
    const message = inject('msg')

(8)向父组件传值

    //子组件 
   // 向父组件传值
<script setup>

        // 向父组件传值
    const emit = defineEmits(['add'])
    const addList = ()=>{
        emit('add', todoVal.value)   //传值时需要使用 .value
        todoVal.value = ''
    }
    const    delAll = () => {
        emit('deleteAll')
    }
    
</script>

<template>
    <header>
        <span class="submit" @click="addList">提交</span>
        <span class="clear" @click="delAll">清空</span>
    </header>
</template>

//父组件
<script setup>
    // 避免使用this   常量修改值时, 需要通过 .value来修改, 在显示时不需要.value
    const  addList = (todoVal)=> {
            if (todoVal=== '') return alert('输入内容不能为空!!!') 
            obj.todoList.unshift({
                // id : `${new Date().getTime()}`,
                todoName: todoVal,
                done: false
            });
    }
    // 删除单个
    const delOne = index =>{
        obj.todoList.splice(index,1)
    }
    //删除全部
    const delAll = ()=>{
        // 从索引为0的开始全删掉   如果使用直接让数组=[]的方式,只是清空了数组, 地址没有变, 无法触发watch监听      使用todoList.value.splice(0) 或 todoList.value.length=0的方式清空    todoList如果是数组需要 .value, 对象不需要写
        obj.todoList.splice(0)
    }
</script>
<template>
    <ToDoHeader @add="addList" @deleteAll="delAll"> </ToDoHeader>
</template>

猜你喜欢

转载自blog.csdn.net/qq_54379580/article/details/129170100