Vue3 - 小兔仙 - day1

课程传送门Day1-02.认识Vue3_哔哩哔哩_bilibiliDay1-02.认识Vue3是黑马程序员前端Vue3小兔鲜电商项目实战,vue3全家桶从入门到实战电商项目一套通关的第2集视频,该合集共计106集,视频收藏或关注UP主,及时了解更多相关视频内容。icon-default.png?t=O83Ahttps://www.bilibili.com/video/BV1Ac411K7EQ?spm_id_from=333.788.player.switch&vd_source=47069034f15b735820208ba3859783b5&p=2

0.学习大纲

1.认识Vue3

Vue3初体验

组合式API体验:通过 Counter 案例,体验 Vue3 新引入的组合式API

使用Vue2写代码

<script>
export default {
  data(){
    return {
      count:0
    }
  },
  methods:{
    addCount(){
      this.count++
    }
  }
}
</script>

Vue3 的组合式API 示例

<script setup>
import { ref } from 'vue'
const count = ref(0)
const addCount = ()=> count.value++
</script>

特点:

  1. 代码量变少

  2. 分散式维护变成集中式维护

Vue3的更多优势

2.使用create-vue搭建Vue3项目

认识create-vue

create-vue是Vue官方新的脚手架工具,底层切换到了 vite (下一代前端工具链,之前vue2底层用的是 webpack),为开发提供极速响应。

使用create-vue创建项目

前置条件 - 已安装16.0或更高版本的Node.js

执行如下命令,这一指令将会安装并执行 create-vue

npm init vue@latest

依次执行命令

3.熟悉项目和关键文件

4.组合式API - setup选项

setup选项的写法和执行时机

执行时机

在beforeCreate钩子之前执行

检验

<script>
export default {
  setup() {
    console.log("setup");
  },
  beforeCreate() {
    console.log("beforeCreate");
  },
};
</script>

 控制台打印

setup语法糖

代码示例

<script setup>
  const message = "Hello Vue 3 + Vite";
  const printMessage = () => {
    console.log(message);
  }
</script>

<template>
  <div>
    <h1>{{ message }}</h1>
    <button @click="printMessage">Print Message</button>
  </div>
</template>

5.组合式API - reactive和ref函数

相当于 vue2 中的 data 函数

reactive

接受 对象类型数据 的参数传入并返回一个响应式的对象

代码示例

<script setup>
// 1.导入函数
import { reactive } from "vue";

// 2.使用函数
const state = reactive({
  count: 0,
});
const addCount = () => {
  state.count++;
};
</script>

<template>
  <div>
    <h1>{{ state.count }}</h1>
    <button @click="addCount">Add Count</button>
  </div>
</template>

ref

接收 简单类型或者对象类型 的数据传入并返回一个响应式的对象

<script setup>
 // 导入
 import { ref } from 'vue'
 // 执行函数 传入参数 变量接收
 const count = ref(0)
 const setCount = ()=>{
   // 修改数据更新视图必须加上.value
   count.value++
 }
</script>

<template>
  <button @click="setCount">{{count}}</button>
</template>

 reactive 对比 ref

  1. 都是用来生成响应式数据

  2. 不同点

    1. reactive不能处理简单类型的数据

    2. ref参数类型支持更好,但是必须通过.value做访问修改

    3. ref函数内部的实现依赖于reactive函数

  3. 在实际工作中的推荐使用ref函数,减少记忆负担

6.组合式API - computed

计算属性基本思想和Vue2保持一致,组合式API下的计算属性只是修改了API写法

<script setup>
import { ref, computed } from "vue";

const list = ref([1, 2, 3]);
const newList = computed(() => {
  return list.value.map((item) => item * 2);
});
</script>

<template>
  <div>
    <h1>这是原来的数组:{{ list }}</h1>
    <h1>这是计算属性:{{ newList }}</h1>
  </div>
</template>

计算属性使用场景

1.计算属性中不应该有“副作用”

        如 异步请求/修改dom

2.避免直接修改计算属性的值

        计算属性应该是只读的

7.组合式API - watch

侦听一个或者多个数据的变化,数据变化时执行回调函数,俩个额外参数 immediate控制立刻执行,deep开启深度侦听

侦听单个数据

<script setup>
  // 1. 导入watch
  import { ref, watch } from 'vue'
  const count = ref(0)
  // 2. 调用watch 侦听变化
  watch(count, (newValue, oldValue)=>{
    console.log(`count发生了变化,老值为${oldValue},新值为${newValue}`)
  })
</script>

侦听多个数据

 侦听多个数据,第一个参数可以改写成数组的写法

<script setup>
  // 1. 导入watch
  import { ref, watch } from 'vue'
  const count = ref(0)
  const name = ref('cp')
  // 2. 调用watch 侦听变化
  watch([count, name], ([newCount, newName],[oldCount,oldName])=>{
    console.log(`count或者name变化了,[newCount, newName],[oldCount,oldName])
  })
</script>

immediate

在侦听器创建时立即出发回调,响应式数据变化之后继续执行回调。一般在用户没有填写任何数据之前,先默认请求后端获取一次数据。

<script setup>
  // 1. 导入watch
  import { ref, watch } from 'vue'
  const count = ref(0)
  // 2. 调用watch 侦听变化
  watch(count, (newValue, oldValue)=>{
    console.log(`count发生了变化,老值为${oldValue},新值为${newValue}`)
  },{
    immediate: true
  })
</script>

 deep

通过watch监听的ref对象默认是浅层侦听的,直接修改嵌套的对象属性不会触发回调执行,需要开启deep

<script setup>
  // 1. 导入watch
  import { ref, watch } from 'vue'
  const state = ref({ count: 0 })
  // 2. 监听对象state
  watch(state, ()=>{
    console.log('数据变化了')
  })
  const changeStateByCount = ()=>{
    // 直接修改不会引发回调执行
    state.value.count++
  }
</script>

<script setup>
  // 1. 导入watch
  import { ref, watch } from 'vue'
  const state = ref({ count: 0 })
  // 2. 监听对象state 并开启deep
  watch(state, ()=>{
    console.log('数据变化了')
  },{deep:true})
  const changeStateByCount = ()=>{
    // 此时修改可以触发回调
    state.value.count++
  }
</script>

deep有性能损耗,尽量不开启。

8.组合式API - 生命周期函数

选项式对比组合式

生命周期函数基本使用

  1. 导入生命周期函数

  2. 执行生命周期函数,传入回调

<scirpt setup>
import { onMounted } from 'vue'
onMounted(()=>{
  // 自定义逻辑
})
</script>

执行多次

生命周期函数执行多次的时候,会按照顺序依次执行

<script setup>
import { onMounted } from "vue";
onMounted(() => {
  console.log("onMounted1"); 
});

onMounted(() => {
  console.log("onMounted2");
});
</script>

9.组合式API - 父子通信

父传子 defineProps

基本思想

  1. 父组件中给子组件绑定属性

  2. 子组件内部通过props选项接收数据

子传父 defineEmits

基本思想

  1. 父组件中给子组件标签通过@绑定事件

  2. 子组件内部通过 emit 方法触发事件

10.组合式API - 模版引用

概念:通过 ref 标识 获取真实的 dom对象或者组件实例对象

使用场景示例:修改一行数据,弹出修改子组件页面,获取自组件的引用,调用子组件的方法

基本使用

实现步骤:

  1. 调用ref函数生成一个ref对象

  2. 通过ref标识绑定ref对象到标签

绑定dom元素

绑定组件

defineExpose

默认情况下在 <script setup>语法糖下组件内部的属性和方法是不开放给父组件访问的,可以通过defineExpose编译宏指定哪些属性和方法容许访问 说明:指定testMessage属性可以被访问到。

11.组合式API - provide和inject

作用和场景

顶层组件向任意的底层组件传递数据和方法,实现跨层组件通信

跨层传递普通数据

实现步骤

  1. 顶层组件通过 provide 函数提供数据

  2. 底层组件通过 inject 函数提供数据

跨层传递响应式数据

在调用provide函数时,第二个参数设置为ref对象

跨层传递方法

顶层组件可以向底层组件传递方法,底层组件调用方法修改顶层组件的数据

12.Vue3综合小案例

项目地址

git clone  http://git.itcast.cn/heimaqianduan/vue3-basic-project.git

项目说明

  1. 模版已经配置好了案例必须的安装包

  2. 案例用到的接口在 README.MD文件 中

  3. 案例项目有俩个分支,main主分支为开发分支,complete分支为完成版分支供开发完参考

项目视图

列表渲染功能实现

<script setup>
import Edit from "./components/Edit.vue";
import axios from "axios";
import { onMounted, ref } from "vue";

// 列表渲染
// 实现思路:1.定义一个响应式列表 -> 2.调用接口,接收响应数据 -> 3.将接口的响应式数据放到定义的响应式列表中 -> 4.渲染列表
const list = ref([]);
const getList = async () => {
  const res = await axios.get("/list");
  list.value = res.data;
};
onMounted(() => {
  getList();
});

// TODO: 删除功能

// TODO: 编辑功能
</script>

<template>
  <div class="app">
    <el-table :data="list">
      <el-table-column label="ID" prop="id"></el-table-column>
      <el-table-column label="姓名" prop="name" width="150"></el-table-column>
      <el-table-column label="籍贯" prop="place"></el-table-column>
      <el-table-column label="操作" width="150">
        <template #default>
          <el-button type="primary" link>编辑</el-button>
          <el-button type="danger" link>删除</el-button>
        </template>
      </el-table-column>
    </el-table>
  </div>
  <Edit />
</template>

<style scoped>
.app {
  width: 980px;
  margin: 100px auto 0;
}
</style>

删除功能实现

<script setup>
import Edit from "./components/Edit.vue";
import axios from "axios";
import { onMounted, ref } from "vue";

// 列表渲染
// 实现思路:1.定义一个响应式列表 -> 2.调用接口,接收响应数据 -> 3.将接口的响应式数据放到定义的响应式列表中 -> 4.渲染列表
const list = ref([]);
const getList = async () => {
  const res = await axios.get("/list");
  list.value = res.data;
};
onMounted(() => {
  getList();
});

// 删除功能
// 实现思路:1.给删除按钮绑定点击事件 -> 2.点击事件中调用接口,将需要删除的数据id传给接口 -> 3.接口返回成功后,重新调用接口,刷新列表
const del = async (id) => {
  await axios.delete(`/del/${id}`);
  getList();
};

// TODO: 编辑功能
</script>

<template>
  <div class="app">
    <el-table :data="list">
      <el-table-column label="ID" prop="id"></el-table-column>
      <el-table-column label="姓名" prop="name" width="150"></el-table-column>
      <el-table-column label="籍贯" prop="place"></el-table-column>
      <el-table-column label="操作" width="150">
        <template #default="{ row }">
          <el-button type="primary" link>编辑</el-button>
          <el-button type="danger" link @click="del(row.id)">删除</el-button>
        </template>
      </el-table-column>
    </el-table>
  </div>
  <Edit />
</template>

<style scoped>
.app {
  width: 980px;
  margin: 100px auto 0;
}
</style>

编辑功能实现

<script setup>
import Edit from "./components/Edit.vue";
import axios from "axios";
import { onMounted, ref } from "vue";

// 列表渲染
// 实现思路:1.定义一个响应式列表 -> 2.调用接口,接收响应数据 -> 3.将接口的响应式数据放到定义的响应式列表中 -> 4.渲染列表
const list = ref([]);
const getList = async () => {
  const res = await axios.get("/list");
  list.value = res.data;
};
onMounted(() => {
  getList();
});

// 删除功能
// 实现思路:1.给删除按钮绑定点击事件 -> 2.点击事件中调用接口,将需要删除的数据id传给接口 -> 3.接口返回成功后,重新调用接口,刷新列表
const del = async (id) => {
  await axios.delete(`/del/${id}`);
  getList();
};

// 编辑功能
// 实现思路:1.打开弹窗 -> 2.回填数据 -> 3.点击确定按钮,调用接口,将编辑好的数据传给接口 -> 4.接口返回成功后,重新调用接口,刷新列表
const editRef = ref(null);
const openDialog = (row) => {
  editRef.value.openDialog(row);
};
</script>

<template>
  <div class="app">
    <el-table :data="list">
      <el-table-column label="ID" prop="id"></el-table-column>
      <el-table-column label="姓名" prop="name" width="150"></el-table-column>
      <el-table-column label="籍贯" prop="place"></el-table-column>
      <el-table-column label="操作" width="150">
        <template #default="{ row }">
          <el-button type="primary" link @click="openDialog(row)">编辑</el-button>
          <el-button type="danger" link @click="del(row.id)">删除</el-button>
        </template>
      </el-table-column>
    </el-table>
  </div>
  <Edit ref="editRef" @on-update="getList"/>
</template>

<style scoped>
.app {
  width: 980px;
  margin: 100px auto 0;
}
</style>
<script setup>
// TODO: 编辑
import { ref } from "vue";
import axios from "axios";
// 弹框开关
const dialogVisible = ref(false);
const form = ref({
  id: "",
  name: "",
  place: "",
});

const openDialog = (row) => {
  dialogVisible.value = true;
  form.value.id = row.id;
  form.value.name = row.name;
  form.value.place = row.place;
};

const emit = defineEmits(["on-update"]);
const onUpdate = () => {
  // 调用更新接口
  axios.patch(`/edit/${form.value.id}`, {
    name: form.value.name,
    place: form.value.place,
  });
  // 关闭弹框
  dialogVisible.value = false;
  // 通知父组件刷新列表
  emit("on-update");
};

defineExpose({
  openDialog,
});
</script>

<template>
  <el-dialog v-model="dialogVisible" title="编辑" width="400px">
    <el-form label-width="50px">
      <el-form-item label="姓名">
        <el-input placeholder="请输入姓名" v-model="form.name" />
      </el-form-item>
      <el-form-item label="籍贯">
        <el-input placeholder="请输入籍贯" v-model="form.place" />
      </el-form-item>
    </el-form>
    <template #footer>
      <span class="dialog-footer">
        <el-button @click="dialogVisible = false">取消</el-button>
        <el-button type="primary" @click="onUpdate">确认</el-button>
      </span>
    </template>
  </el-dialog>
</template>

<style scoped>
.el-input {
  width: 290px;
}
</style>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值