【vue3】入门

vue3官网 https://cn.vuejs.org/guide/quick-start.html

1. 创建一个项目

前提条件:已经安装18.3以上版本的node.js环境。
打开终端输入命令


image.png

2. 项目结构

image.png

3. 响应式

vue2使用defineProperty来为每个属性定义get和set方法。当属性被访问时,get方法会被调用。当属性变化,set会被调用,通知观察者更新视图。
vue3使用proxy来实现响应式,proxy是es6新特性,为了对目标的作用主要是通过handler对象中的拦截方法拦截目标对象target的某些行为(如属性查找、赋值、枚举、函数调用等)

  1. ref 官网推荐使用ref来定义属性,ref具有深层响应式。也就是改变嵌套的对象或者数组时,变化也将会被检测到。注:使用时要使用 .value 。在模板和插值语法中将会被自动结构不需要.value。即 {{ boj.b }}
const obj = ref( {a: [ a1: 2, a2: 3], b: 'bbb'} );
console.log(obj.value.b);  // 'bbb'
  1. reactive 使用时不需要写.value。返回的对象直接是proxy对象。
    注:不支持原始类型的响应式(如:string、number 或 boolean)。不能直接赋值会丢失响应式。解构后也将丢失响应式。具体可以看官网例子
const obj = reactive( {a: [ a1: 2, a2: 3], b: 'bbb'} );
console.log(obj.b);  // 'bbb'

4. 计算属性 computed

计算属性默认是只读的(官网中有可以修改的示例。但需要避免直接修改计算属性值)。和方法的区别是方法每次重新渲染都会重新计算,而计算属性类似一个响应式对象在模板中使用,当内部响应式数据变更时才会计算并存起来。即计算属性值会基于其响应式依赖被缓存

const cont1 = ref( 5 );
const cont2 = ref( 3 );
const sum = computed(()=>{
      return cont1.value +  cont2.value
})

5. 侦听器 watch()和watchEffect()

watch 必须声明需要监听的对象

const x = ref(0)
const y = ref(0)

// 单个 ref
watch(x, (newX) => {
  console.log(`x is ${newX}`)
})

// getter 函数 注意监听响应式对象的属性也需要用函数 () => obj.count,
watch(
  () => x.value + y.value,
  (sum) => {
    console.log(`sum of x + y is: ${sum}`)
  }
)
// 监听响应式对象的属性
watch(
  () => obj.count,
  (val) => {
    console.log(obj.count)
  }
)

// 多个来源组成的数组
watch([x, () => y.value], ([newX, newY]) => {
  console.log(`x is ${newX} and y is ${newY}`)
})

watchEffect 在执行期间,它会自动追踪 todoId.value 作为依赖(和计算属性类似)。每当 todoId.value 变化时,回调会再次执行。有了 watchEffect(),我们不再需要明确传递 todoId 作为源值

const todoId = ref(1);
const data = ref(null);

watchEffect(async () => {
  const response = await fetch(
    `https://jsonplaceholder.typicode.com/todos/${todoId.value}`
  )
  data.value = await response.json();
})

注意:watch和watchEffect的区别:watch需要声明监听的对象,watchEffect只跟踪回调中被使用到的的响应式依赖

6. 生命周期

image.png

7. defineProps 组件声明可以接受的参数。props拥有单向数据流的特点,子组件不可以修改父组件props中的值。

<script setup>
const props = defineProps({
 // 基础类型检查
  // (给出 `null` 和 `undefined` 值则会跳过任何类型检查)
  propA: Number,
  // 多种可能的类型
  propB: [String, Number],
  // 必传,且为 String 类型
  propC: {
    type: String,
    required: true
  },
  // 必传但可为 null 的字符串
  propD: {
    type: [String, null],
    required: true
  },
  // Number 类型的默认值
  propE: {
    type: Number,
    default: 100
  },
  // 对象类型的默认值
  propF: {
    type: Object,
    // 对象或数组的默认值
    // 必须从一个工厂函数返回。
    // 该函数接收组件所接收到的原始 prop 作为参数。
    default(rawProps) {
      return { message: 'hello' }
    }
  },
  // 自定义类型校验函数
  // 在 3.4+ 中完整的 props 作为第二个参数传入
  propG: {
    validator(value, props) {
      // The value must match one of these strings
      return ['success', 'warning', 'danger'].includes(value)
    }
  },
  // 函数类型的默认值
  propH: {
    type: Function,
    // 不像对象或数组的默认,这不是一个
    // 工厂函数。这会是一个用来作为默认值的函数
    default() {
      return 'Default function'
    }
  }
  })
</script>

8. defineModel是一个便利宏(vue3.4+)

编译器将其展开为以下内容:
一个名为 modelValue 的 prop,本地 ref 的值与其同步;
一个名为 update:modelValue 的事件,当本地 ref 的值发生变更时触发。

3.4版本前

// 父组件
<Child
  :modelValue="foo"
  @update:modelValue="$event => (foo = $event)"
/>

// 子组件
<script setup>
const props = defineProps(['modelValue'])
const emit = defineEmits(['update:modelValue'])
</script>

<template>
  <input
    :value="props.modelValue"
    @input="emit('update:modelValue', $event.target.value)"
  />
</template>

3.4版本后

// 声明 "modelValue" prop,由父组件通过 v-model 使用
const model = defineModel()
// 或者:声明带选项的 "modelValue" prop
const model = defineModel({ type: String })

// 在被修改时,触发 "update:modelValue" 事件
model.value = "hello"

// 声明 "count" prop,由父组件通过 v-model:count 使用
const count = defineModel("count")
// 或者:声明带选项的 "count" prop
const count = defineModel("count", { type: Number, default: 0 })

function inc() {
  // 在被修改时,触发 "update:count" 事件
  count.value++
}
image.png
  1. provide 和 inject
    一个父组件相对于其所有的后代组件,会作为依赖提供者。任何后代的组件树,无论层级有多深,都可以注入由父组件提供给整条链路的依赖。
// 父组件provide传值
<script setup>
import { provide } from 'vue'
provide(/* 注入名 */ 'message', /* 值 */ 'hello!')
</script>

// 后代组件inject接收
<script setup>
  import {inject} from 'vue'
  inject('message');
</script>
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容