1.组件的三大组成部分注意点(结构/样式/逻辑)scoped解决样式冲突/data是一个函数2.组件通信组件通信语法父传子子传父

news2025/1/14 11:48:08

学习目标

1.组件的三大组成部分注意点(结构/样式/逻辑)

scoped解决样式冲突/data是一个函数

2.组件通信

  1. 组件通信语法
  1. 父传子
  1. 子传父
  1. 非父子通信(扩展)

3.综合案例:小黑记事本(组件版)

  1. 拆分组件
  1. 列表渲染
  1. 数据添加
  1. 数据删除
  1. 列表统计
  1. 清空
  1. 持久化

4.进阶语法

  1. v-model原理
  1. v-model应用于组件
  1. sync修饰符
  1. ref和$refs
  1. $nextTick

组件的三大组成部分注意点

template只能有一个根元素

约束:.vue文件中的template中如果写了两个元素,则会报如下错误

解决:保证template中只有一个根元素即可

scoped解决样式冲突

  1. 全局样式: 默认组件中的样式会作用到全局,任何一个组件中都会受到此样式的影响
  2. 局部样式: 可以给组件加上scoped 属性,可以让样式只作用于当前组件

默认情况:写在组件中的样式会 全局生效 → 因此很容易造成多个组件之间的样式冲突问题。

解决:在组件style标签上增加scoped来解决

代码演示
<template>
  <div class="base-one">
    BaseOne
  </div>

</template>

<script>
export default {

}
</script>

<!-- 增加了scoped,表示局部样式,不会被覆盖  -->
<style scoped>
  .base-one{
    color:red;
  }
</style>
<template>
  <div class="base-one">
    BaseTwo
  </div>

</template>

<script>
export default {

}
</script>

<!-- 增加了scoped,表示局部样式,不会被覆盖  -->
<style scoped>
  .base-one{
    color:green;
  }
</style>
<template>
  <div id="app">
    <BaseOne></BaseOne>

    <BaseTwo></BaseTwo>

  </div>

</template>

<script>
import BaseOne from './components/BaseOne'
import BaseTwo from './components/BaseTwo'
export default {
  name: 'App',
  components: {
    BaseOne,
    BaseTwo
  }
}
</script>
scoped原理
  1. 当前组件内标签都被添加data-v-hash值 的属性 ,每个组件的hash值是不同的
  1. css选择器都被添加 [data-v-hash值] 的属性选择器

最终效果: 必须是当前组件的元素, 才会有这个自定义属性, 才会被这个样式作用到

data必须是一个函数

一个.vue组件的 data 选项必须是一个函数。否则会报错

这么要求的原因:保证每个组件实例,维护独立的一份数据对象,保证组件实例之间的数据相互隔离不受影响

每次创建新的组件实例,都会新执行一次data 函数,得到一个新对象。

代码演示
<template>
  <div class="base-count">
    <button @click="count--">-</button>

    <span>{{ count }}</span>

    <button @click="count++">+</button>

  </div>

</template>

<script>
export default {
  data: function () {
    return {
      count: 100,
    }
  },
}
</script>

<style>
.base-count {
  margin: 20px;
}
</style>
<template>
  <div class="app">
    <BaseCount></BaseCount>
    <BaseCount></BaseCount>
  </div>

</template>

<script>
import BaseCount from './components/BaseCount'
export default {
  components: {
    BaseCount,
  },
}
</script>

<style>
</style>

组件通信

什么是组件通信?组件通信,就是指组件与组件之间的数据传递

为什么要有组件通信?

  • 组件的数据是独立的,无法直接访问其他组件的数据。
  • 想使用其他组件的数据,就需要组件通信

组件之间如何通信?

组件关系分类:① 父子关系 ② 非父子关系

组件通信解决方案:

父子通信流程

  1. 父->子:父组件通过 props 将数据传递给子组件
  2. 子->父:子组件利用 $emit 通知父组件修改数据

父向子通信代码示例

父向子传值步骤:

  1. 准备一个父组件 App.vue,一个子组件Son.vue
  2. 在App.vue中使用Son.vue让它们构成一个父子组件关系,在使用子组件的同时
    1. 通过 :自定义名字="需要传递的值" 将父组件中的数据传给子组件
  1. 子组件内部通过props接收 props:['父组件中自定义名字']
  2. 子组件内部模板中直接使用 props接收的值 {{ 父组件中自定义名字 }}

✨✨ 注意点:父组件中的响应式数据改变,会自动同步到子组件

父组件通过props将数据传递给子组件代码演示

<template>
  <div class="box">
    <!-- 2. 调用子组件的同时 传递数据 -->
    <SonVue 
      :title="msg" 
      :age="age"
      :nums="nums">
      
      </SonVue>
  </div>
</template>

<script>
import SonVue from './components/Son.vue';


export default {
  data(){
    return {
      // 1. 准备需要传递的数据

      /*
      ✨✨ 注意点:
       1. 父组件中的响应式数据改变,会自动同步到子组件
      */ 
      msg:'学前端+鸿蒙来黑马',
      age:10,
      nums:[1,2,3]
    }
  },
  // 局部注册组件
  components: {
    SonVue
  },
};
</script>

<style>
</style>
<template>
  <div class="son" style="border:3px solid #000;margin:10px">
    <!-- 4. 显示数据-->
    我是Son组件 {{ title }}{{ age }}{{ nums }}
  </div>

</template>

<script>
export default {
  name: 'Son-Child',
//   3. 接收父组件传入的title属性的值
  props:['title','age','nums']
}
</script>

<style>

</style>
子向父通信代码示例

子组件利用 $emit 将自己的数据传递给父组件

子向父传值步骤:

  1. $emit触发事件,给父组件发送消息通知
  2. 父组件监听$emit触发的事件
  3. 提供处理函数,在函数的形参中获取传过来的参数

注意:上面代码其实是一个 父子通信的双向数据绑定

<template>
  <div class="box">
    <div>App.vue-> 父组件</div>
    <SonVue></SonVue>
  </div>
</template>

<script>
import SonVue from "./components/Son.vue";

export default {
  data() {
    return {};
  },
  // 局部注册组件
  components: {
    SonVue,
  },
};
</script>

<style>
.box {
   height: 300px;
   width: 300px;
   background: pink;
   display: flex;
   align-content: center;
   flex-direction: column;
}
</style>
<template>
  <div class="son" style="border:3px solid #000;margin:10px;height:80px;width:150px">
    <!-- 4. 显示数据-->
    子组件
  </div>

</template>

<script>
export default {
  name: 'Son-Child',

}
</script>

<style>

</style>

随堂演示代码:

<template>
  <div class="box">
    <div>App.vue-> 父组件 {{ appTile }}</div>
    <!-- 1. 第一步:在使用子组件的时候注册一个自定义事件
    此处的事件名:changeTitle
     -->
    <SonVue @changeTitle="changeHander"></SonVue>
  </div>
</template>

<script>
import SonVue from "./components/Son.vue";

export default {
  data() {
    return {
      appTile:'鸿蒙3期'
    };
  },
  // 局部注册组件
  components: {
    SonVue,
  },
  methods:{
    changeHander(title){
      console.log(title);      
      this.appTile = title
    }
  }
};
</script>

<style>
.box {
   height: 300px;
   width: 300px;
   background: pink;
   display: flex;
   align-content: center;
   flex-direction: column;
}
</style>
<template>
  <div class="son" style="border:3px solid #000;margin:10px;height:80px;width:150px">

    <button @click="changeFn">给父组件传值</button>
  </div>

</template>

<script>
export default {
  name: 'Son-Child',
  methods:{
    changeFn(){
      // 2. 子组件中通过 $emit(父组件调用子组件时注册的那个自定义事件名称,传递给父组件的数据(数据类型任意))
      // 触发父组件注册的自定义事件
      this.$emit('changeTitle','鸿蒙4期')
    }
  }
}
</script>

<style>

</style>

什么是props

Props 定义:组件上 注册的一些 自定义属性

Props 作用:向子组件传递数据

特点:

  1. 可以 传递 任意数量 的prop
  2. 可以 传递 任意类型 的prop

代码演示
<template>
  <div class="app">
    <UserInfo
      :username="username"
      :age="age"
      :isSingle="isSingle"
      :car="car"
      :hobby="hobby"
    ></UserInfo>

  </div>

</template>

<script>
import UserInfo from './components/UserInfo.vue'
export default {
  data() {
    return {
      username: '小帅',
      age: 28,
      isSingle: true,
      car: {
        brand: '宝马',
      },
      hobby: ['篮球', '足球', '羽毛球'],
    }
  },
  components: {
    UserInfo,
  },
}
</script>

<style>
</style>
<template>
  <div class="userinfo">
    <h3>我是个人信息组件</h3>

    <div>姓名:</div>

    <div>年龄:</div>

    <div>是否单身:</div>

    <div>座驾:</div>

    <div>兴趣爱好:</div>

  </div>

</template>

<script>
export default {
  
}
</script>

<style>
.userinfo {
  width: 300px;
  border: 3px solid #000;
  padding: 20px;
}
.userinfo > div {
  margin: 20px 10px;
}
</style>

props校验

思考:组件的props数据类型可以乱传吗?不能

  • 比如进度条百分数只能是数字

props校验:为组件的 prop 指定验证要求,不符合要求,控制台就会有错误提示 → 帮助开发者,快速发现错误

props校验的类型:

  • 类型校验(常用)
  • 非空校验
  • 默认值
  • 自定义校验

语法:

① 只校验类型【常用】

② 完整写法

// ✨✨✨注意点:属性的类型必须使用大写 Number,Boolean,String,Array,Object

props校验类型代码演示
<template>
  <div class="app">
    <BaseProgress :w="width"></BaseProgress>

  </div>

</template>

<script>
import BaseProgress from './components/BaseProgress.vue'
export default {
  data() {
    return {
      width: 30,
    }
  },
  components: {
    BaseProgress,
  },
}
</script>

<style>
</style>
<template>
  <div class="base-progress">
    <div class="inner" :style="{ width: w + '%' }">
      <span>{{ w }}%</span>

    </div>

  </div>

</template>

<script>
export default {
  props: ['w'],
}
</script>

<style scoped>
.base-progress {
  height: 26px;
  width: 400px;
  border-radius: 15px;
  background-color: #272425;
  border: 3px solid #272425;
  box-sizing: border-box;
  margin-bottom: 30px;
}
.inner {
  position: relative;
  background: #379bff;
  border-radius: 15px;
  height: 25px;
  box-sizing: border-box;
  left: -3px;
  top: -2px;
}
.inner span {
  position: absolute;
  right: 0;
  top: 26px;
}
</style>

props校验完整写法
props: {
  校验的属性名: {
    type: 类型,  // Number String Boolean ...
    required: true, // 是否必填
    default: 默认值, // 默认值
    validator (value) {
      // 自定义校验逻辑
      return 是否通过校验
    }
  }
},
<script>
export default {
  // 完整写法(类型、默认值、非空、自定义校验)
  props: {
    w: {
      type: Number,
      //required: true,
      default: 0,
      validator(val) {
        // console.log(val)
        if (val >= 100 || val <= 0) {
          console.error('传入的范围必须是0-100之间')
          return false
        } else {
          return true
        }
      },
    },
  },
}
</script>
  1. default和required一般不同时写(因为当时必填项时,肯定是有值的)
  2. default后面如果是简单类型的值,可以直接写默认。如果是复杂类型的值,则需要以函数的形式return一个默认值

props&data、单向数据流

共同点:一个组件中props和data,都可以给组件提供数据

区别:

  • data 的数据是自己的 → 随便改
  • prop 的数据是外部的 → 不能直接改,要遵循 单向数据流

单向数据流:父级props 的数据更新,会向下流动,影响子组件。这个数据流动是单向的

特点:子组件修改prop数据不会影响到父组件的数据

约定:谁的数据,谁负责修改

代码演示
<template>
  <div class="app">
    <BaseCount></BaseCount>

  </div>

</template>

<script>
import BaseCount from './components/BaseCount.vue'
export default {
  components:{
    BaseCount
  },
  data(){
  },
}
</script>

<style>

</style>
<template>
  <div class="base-count">
    <button @click="count--">-</button>

    <span>{{ count }}</span>

    <button @click="count++">+</button>

  </div>

</template>

<script>
export default {
  // 1.自己的数据随便修改  (谁的数据 谁负责)
   data () {
     return {
       count: 100,
     }
   },
  // 2.外部传过来的数据 不能随便修改
  //props: {
  //  count: {
  //    type: Number,
  //  }, 
  //}
}
</script>

<style>
.base-count {
  margin: 20px;
}
</style>

综合案例-小黑记事本组件版

需求:把小黑记事本原有的结构拆成三部分内容:

  1. 头部(TodoHeader)
  2. 列表(TodoMain)
  3. 底部(TodoFooter)
  4. App.vue是它们的父组件,用来统一管理任务数据

功能拆解:

  1. 拆分头部(TodoHeader)、列表(TodoMain)、底部(TodoFooter)三个基础组件 -> App.vue中使用这些基础组件
  2. 列表(TodoMain)组件-显示代办任务 -> App.vue 向 列表(TodoMain)组件 通过props传递任务数组
  3. 列表(TodoMain)组件- 删除代办任务 -> 通过$emit向App.vue中的任务数组中删除数据
  4. 头部(TodoHeader)组件-添加任务 -> 通过$emit向App.vue中的任务数组中追加数据
  5. 底部(TodoFooter)组件- 合计 和 清空功能任务 -> 分别通过 props和$emit来完成
  6. App.vue组件完成任务数据存储 -> watch + localStorage来实现

html,
body {
  margin: 0;
  padding: 0;
}
body {
  background: #fff;
}
button {
  margin: 0;
  padding: 0;
  border: 0;
  background: none;
  font-size: 100%;
  vertical-align: baseline;
  font-family: inherit;
  font-weight: inherit;
  color: inherit;
  -webkit-appearance: none;
  appearance: none;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
}

body {
  font: 14px 'Helvetica Neue', Helvetica, Arial, sans-serif;
  line-height: 1.4em;
  background: #f5f5f5;
  color: #4d4d4d;
  min-width: 230px;
  max-width: 550px;
  margin: 0 auto;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  font-weight: 300;
}

:focus {
  outline: 0;
}

.hidden {
  display: none;
}

.App {
  background: #fff;
  margin: 180px 0 40px 0;
  padding: 15px;
  position: relative;
  box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.2), 0 25px 50px 0 rgba(0, 0, 0, 0.1);
}
.App .header input {
  border: 2px solid rgba(175, 47, 47, 0.8);
  border-radius: 10px;
}
.App .add {
  position: absolute;
  right: 15px;
  top: 15px;
  height: 68px;
  width: 140px;
  text-align: center;
  background-color: rgba(175, 47, 47, 0.8);
  color: #fff;
  cursor: pointer;
  font-size: 18px;
  border-radius: 0 10px 10px 0;
}

.App input::-webkit-input-placeholder {
  font-style: italic;
  font-weight: 300;
  color: #e6e6e6;
}

.App input::-moz-placeholder {
  font-style: italic;
  font-weight: 300;
  color: #e6e6e6;
}

.App input::input-placeholder {
  font-style: italic;
  font-weight: 300;
  color: gray;
}

.App h1 {
  position: absolute;
  top: -120px;
  width: 100%;
  left: 50%;
  transform: translateX(-50%);
  font-size: 60px;
  font-weight: 100;
  text-align: center;
  color: rgba(175, 47, 47, 0.8);
  -webkit-text-rendering: optimizeLegibility;
  -moz-text-rendering: optimizeLegibility;
  text-rendering: optimizeLegibility;
}

.new-todo,
.edit {
  position: relative;
  margin: 0;
  width: 100%;
  font-size: 24px;
  font-family: inherit;
  font-weight: inherit;
  line-height: 1.4em;
  border: 0;
  color: inherit;
  padding: 6px;
  box-shadow: inset 0 -1px 5px 0 rgba(0, 0, 0, 0.2);
  box-sizing: border-box;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
}

.new-todo {
  padding: 16px;
  border: none;
  background: rgba(0, 0, 0, 0.003);
  box-shadow: inset 0 -2px 1px rgba(0, 0, 0, 0.03);
}

.main {
  position: relative;
  z-index: 2;
}

.todo-list {
  margin: 0;
  padding: 0;
  list-style: none;
  overflow: hidden;
}

.todo-list li {
  position: relative;
  font-size: 24px;
  height: 60px;
  box-sizing: border-box;
  border-bottom: 1px solid #e6e6e6;
}

.todo-list li:last-child {
  border-bottom: none;
}

.todo-list .view .index {
  position: absolute;
  color: gray;
  left: 10px;
  top: 20px;
  font-size: 22px;
}

.todo-list li .toggle {
  text-align: center;
  width: 40px;
  /* auto, since non-WebKit browsers doesn't support input styling */
  height: auto;
  position: absolute;
  top: 0;
  bottom: 0;
  margin: auto 0;
  border: none; /* Mobile Safari */
  -webkit-appearance: none;
  appearance: none;
}

.todo-list li .toggle {
  opacity: 0;
}

.todo-list li .toggle + label {
  /*
		Firefox requires `#` to be escaped - https://bugzilla.mozilla.org/show_bug.cgi?id=922433
		IE and Edge requires *everything* to be escaped to render, so we do that instead of just the `#` - https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/7157459/
	*/
  background-image: url('data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23ededed%22%20stroke-width%3D%223%22/%3E%3C/svg%3E');
  background-repeat: no-repeat;
  background-position: center left;
}

.todo-list li .toggle:checked + label {
  background-image: url('data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23bddad5%22%20stroke-width%3D%223%22/%3E%3Cpath%20fill%3D%22%235dc2af%22%20d%3D%22M72%2025L42%2071%2027%2056l-4%204%2020%2020%2034-52z%22/%3E%3C/svg%3E');
}

.todo-list li label {
  word-break: break-all;
  padding: 15px 15px 15px 60px;
  display: block;
  line-height: 1.2;
  transition: color 0.4s;
}

.todo-list li.completed label {
  color: #d9d9d9;
  text-decoration: line-through;
}

.todo-list li .destroy {
  display: none;
  position: absolute;
  top: 0;
  right: 10px;
  bottom: 0;
  width: 40px;
  height: 40px;
  margin: auto 0;
  font-size: 30px;
  color: #cc9a9a;
  margin-bottom: 11px;
  transition: color 0.2s ease-out;
}

.todo-list li .destroy:hover {
  color: #af5b5e;
}

.todo-list li .destroy:after {
  content: '×';
}

.todo-list li:hover .destroy {
  display: block;
}

.todo-list li .edit {
  display: none;
}

.todo-list li.editing:last-child {
  margin-bottom: -1px;
}

.footer {
  color: #777;
  padding: 10px 15px;
  height: 20px;
  text-align: center;
  border-top: 1px solid #e6e6e6;
}

.footer:before {
  content: '';
  position: absolute;
  right: 0;
  bottom: 0;
  left: 0;
  height: 50px;
  overflow: hidden;
  box-shadow: 0 1px 1px rgba(0, 0, 0, 0.2), 0 8px 0 -3px #f6f6f6,
    0 9px 1px -3px rgba(0, 0, 0, 0.2), 0 16px 0 -6px #f6f6f6,
    0 17px 2px -6px rgba(0, 0, 0, 0.2);
}

.todo-count {
  float: left;
  text-align: left;
}

.todo-count strong {
  font-weight: 300;
}

.filters {
  margin: 0;
  padding: 0;
  list-style: none;
  position: absolute;
  right: 0;
  left: 0;
}

.filters li {
  display: inline;
}

.filters li a {
  color: inherit;
  margin: 3px;
  padding: 3px 7px;
  text-decoration: none;
  border: 1px solid transparent;
  border-radius: 3px;
}

.filters li a:hover {
  border-color: rgba(175, 47, 47, 0.1);
}

.filters li a.selected {
  border-color: rgba(175, 47, 47, 0.2);
}

.clear-completed,
html .clear-completed:active {
  float: right;
  position: relative;
  line-height: 20px;
  text-decoration: none;
  cursor: pointer;
}

.clear-completed:hover {
  text-decoration: underline;
}

.info {
  margin: 50px auto 0;
  color: #bfbfbf;
  font-size: 15px;
  text-shadow: 0 1px 0 rgba(255, 255, 255, 0.5);
  text-align: center;
}

.info p {
  line-height: 1;
}

.info a {
  color: inherit;
  text-decoration: none;
  font-weight: 400;
}

.info a:hover {
  text-decoration: underline;
}

/*
	Hack to remove background from Mobile Safari.
	Can't use it globally since it destroys checkboxes in Firefox
*/
@media screen and (-webkit-min-device-pixel-ratio: 0) {
  .toggle-all,
  .todo-list li .toggle {
    background: none;
  }

  .todo-list li .toggle {
    height: 40px;
  }
}

@media (max-width: 430px) {
  .footer {
    height: 50px;
  }

  .filters {
    bottom: 10px;
  }
}
  <!-- 输入框 拷贝进 TodoHeader.vue -->
  <header class="header">
    <h1>小黑记事本</h1>
    <input placeholder="请输入任务" class="new-todo" />
    <button class="add">添加任务</button>
  </header>


  <!-- 列表区域 - 拷贝进TodoMain.vue -->
  <section class="main">
    <ul class="todo-list">
      <li class="todo">
        <div class="view">
          <span class="index">1.</span> <label>吃饭饭</label>
          <button class="destroy"></button>
        </div>
      </li>
    </ul>
  </section>


  <!-- 统计和清空- 拷贝进-TodoFooter.vue -->
  <footer class="footer">
    <!-- 统计 -->
    <span class="todo-count">合 计:<strong> 1 </strong></span>
    <!-- 清空 -->
    <button class="clear-completed">
      清空任务
    </button>
  </footer>

下载后,替换项目中的src目录

src.rar

📎06-小黑记事本组件版-完成TodoMain组件开发.rar

语法进阶

非父子通信-事件总线event bus

作用:事件总线event bus可以用在非父子组件之间,进行简易消息传递 (复杂场景→ Vuex)

需求:B组件向C组件进行传递数据

使用步骤:(媒婆传话)

  1. 创建一个都能访问到的事件总线 (空 Vue 实例) → utils/EventBus.js

  1. C 组件(接收方),监听 Bus 实例的事件

  1. B 组件(发送方),触发 Bus 实例的事件

代码示例
import Vue from 'vue'
const Bus  =  new Vue()
export default Bus
<template>
  <div class="base-a">
    我是C组件(接收方)
    <p>{{msg}}</p>  
  </div>

</template>

<script>

import Bus from '../utils/EventBus.js'

export default {
  data() {
    return {
      msg: '',
    }
  },
  created(){
    Bus.$on('sendMsg',(msg)=>{
        this.msg = msg
    })
  }
}
</script>

<style scoped>
.base-a {
  width: 200px;
  height: 200px;
  border: 3px solid #000;
  border-radius: 3px;
  margin: 10px;
}
</style>
<template>
  <div class="base-b">
    <div>我是B组件(发布方)</div>

    <button @click="sendMsg">发送消息</button>
  </div>
</template>

<script>
//   1. 导入Bus对象(媒人导入)
import Bus from '../utils/EventBus.js'

export default {
  methods: {
    sendMsg() {
      // 发送方:调用Bus上的$emit方法来触发事件     

      // 2. 调用Bus $emit方法触发事件
      Bus.$emit('sendMsg','我是来自B组件的消息')
    },
  },
};
</script>

<style scoped>
.base-b {
  width: 200px;
  height: 200px;
  border: 3px solid #000;
  border-radius: 3px;
  margin: 10px;
}
</style>
<template>
  <div class="app">
    <BaseA></BaseA>

    <BaseB></BaseB> 
  </div>

</template>

<script>
import BaseA from './components/BaseA.vue'
import BaseB from './components/BaseB.vue' 
export default {
  components:{
    BaseA,
    BaseB
  }
}
</script>

<style>

</style>

非父子通信-provide&inject

作用:provide&inject可以实现组件的跨层级共享数据

provide&inject传递数据使用步骤:

  1. 父组件 provide 提供数据

  1. 子/孙组件 inject 取值使用

注意:

  1. provide提供的简单类型的数据不是响应式的,复杂类型数据是响应式。(推荐提供复杂类型数据 ->定义在data()函数中)
  2. 子/孙组件通过inject获取的数据,不能在自身组件内修改
  1. 父组件 provide提供数据
export default {
  provide () {
    return {
       // 普通类型【非响应式】
       color: this.color, 
       // 复杂类型【响应式】
       userInfo: this.userInfo, 
    }
  }
}

2.子/孙组件 inject获取数据

export default {
  inject: ['color','userInfo'],
  created () {
    console.log(this.color, this.userInfo)
  }
}

provide和inject传值静态组件结构.rar

<template>
  <div class="box">
    我是App.vue
    <BaseBVue></BaseBVue>
  </div>
</template>

<script>
import BaseBVue from "./components/BaseB.vue";

export default {
  data() {
    return {
      message: { msg: "我是App.vue中的数据" },
    };
  },
  provide() {
    return {
      msg: this.message,
    };
  },
  components: {
    BaseBVue,
  },
};
</script>

<style scoped>
.box {
  width: 300px;
  height: 300px;
  border: 3px solid #000;
  border-radius: 3px;
  margin: 10px;
}
</style>
<template>
  <div class="base-b">
    <div>
      我是B组件{{msg.msg}}
      <BaseCVue></BaseCVue>
    </div>
  </div>
</template>

<script>
import BaseCVue from "./BaseC.vue";

export default {
  inject:['msg'],
  components: {
    BaseCVue,
  },
};
</script>

<style scoped>
.base-b {
  width: 200px;
  height: 200px;
  border: 3px solid #000;
  border-radius: 3px;
  margin: 10px;
}
</style>
<template>
  <div class="base-c">
    <div>
      我是C组件

      {{msg.msg}}
    </div>
  </div>
</template>

<script>

export default {
  // 通过inject接收数据
  inject:['msg']

};
</script>

<style scoped>
.base-c {
  width: 100px;
  height: 100px;
  border: 3px solid #000;
  border-radius: 3px;
  margin: 10px;
}
</style>

组件传值阶段总结

  1. 父传子
  2. 子传父
  3. 事件总线- 非父子关系
  4. 家族树传值 - provide 和 inject

父子组件双向数据绑定

思考:默认情况下我们可以给组件用上 v-model进行双向数据绑定吗? 不能 -> 那么如何做?

解决方法:

  1. 用vue实现 v-model 的写法(原理)
  2. v-bind:属性.sync 修饰符

v-model原理

v-model原理:v-model本质上是一个语法糖 -> 应用在输入框上,就是value属性input事件 的合写

作用:v-model是双向数据绑定

① 数据变,视图跟着变 :value

② 视图变,数据跟着变 @input

说明:$event 用于在模板中,获取事件的形参 (这里的$event就是事件对象e)

<template>
  <div class="app">
    <!-- v-model实现双向数据绑定  -->
    <input type="text"  v-model="msg" />
    <br /> 
    <!-- v-model的原理写法  -->
    <input type="text" :value="msg" @input="msg = $event.target.value" />
  </div>

</template>

<script>
export default {
  data() {
    return {
      msg1: '',
      msg2: '',
    }
  },
}
</script> 
<style>
</style>

说明:

不同的表单元素, v-model在底层的处理机制不一样。

  • 给文本框,文本域 -> Vue框架底层是拆解成 value属性 + input事件来实现
  • 下拉框 -> Vue框架底层是拆解成 value属性 + change事件来实现
  • 给复选框,单选框 使用v-model ->Vue框架底层是拆解成 checked属性和change事件 来实现
<template>
  <div class="app">
    <!-- v-model实现双向数据绑定  -->
    <input type="checkbox"  v-model="isSelected" />
    <br /> 
    <!-- v-model的原理写法  -->
   <input type="checkbox" :checked="isSelected" @change="isSelected = $event.target.checked" >
  </div>

</template>

<script>
export default {
  data() {
    return {
     isSelected:true
    }
  },
}
</script> 
<style>
</style>
<template>
  <div>
     <!-- v-model实现双向数据绑定  -->
    <select v-model="cid">
      <option value="101">北京</option>
      <option value="102">上海</option>
    </select>
    
     <!-- v-model的原理写法  -->
    <select :value="cid" @change="cid = $event.target.value">
      <option value="101">北京</option>
      <option value="102">上海</option>
    </select>
  </div>

</template>

<script>
export default {
  data(){
  return {
    cid:'102'
  }
  }
}
</script>

<style>
</style>
案例-封装城市下拉组件

需求:使用v-model实现子组件(BaseSelect.vue)和父组件(App.vue)数据的双向绑定

  • App.vue中定义城市id,传递给BaseSelect.vue后,自动选择对应的城市
  • 用户重新选择城市后,将新的城市id回传给父组件App.vue

拆解:

  1. 创建子组件BaseSelect.vue 静态结构
  2. 在父组件App.vue中使用子组件BaseSelect的同时,通过v-model传入城市id 102
  3. BaseSelect.vue中通过props:{value:String}接收传入的城市id,并将value绑定在select元素上
  4. BaseSelect.vue中通过在select标签上注册@change事件,结合 this.$emit('input', e.target.value)将新选择的城市id回传给App.vue
<template>
  <div class="app">
    <BaseSelect></BaseSelect>
  </div>
</template>
<script>
import BaseSelect from './components/BaseSelect.vue'
export default {
  data() {
    return {
      selectId: '102',
    }
  },
  components: {
    BaseSelect,
  },
}
</script>

<style>
</style>
<template>
  <div>
    <select>
      <option value="101">北京</option>

      <option value="102">上海</option>

      <option value="103">武汉</option>

      <option value="104">广州</option>

      <option value="105">深圳</option>

    </select>

  </div>

</template>

<script>
export default {
}
</script>

<style>
</style>

实现的核心代码:

<template>
  <div>
    <select v-model="value" @change="cityChange">
      <option value="101">北京</option>

      <option value="102">上海</option>

      <option value="103">武汉</option>

      <option value="104">广州</option>

      <option value="105">深圳</option>
    </select>
  </div>
</template>

<script>
export default {
  methods: {
    cityChange() {
      // console.log(this.value);
      this.$emit('input',this.value)
    },
  },
  props: {
    value: {
      type: String,
    },
  },
};
</script>

<style>
</style>
<template>
  <div>
    App.vue组件 {{selectId}}
    <hr />
    <!-- v-model双向绑定  -->
    <BaseSelect v-model="selectId"></BaseSelect>

      <!-- v-model双向绑定实现原理  -->
    <BaseSelect :value="selectId" @input="inputHander"></BaseSelect>

    <!-- <input type="text" v-model="selectId">

    <input type="text" :value="selectId" @input="selectId = $event.target.value">

     -->
  </div>
</template>
<script>
import BaseSelect from "./components/BaseB.vue";
export default {
  data() {
    return {
      selectId: "102",
    };
  },
  methods: {
    inputHander(cityId) {
      this.selectId = cityId
    },
  },
  components: {
    BaseSelect,
  },
};
</script>

<style scoped>
</style>

.sync修饰符

作用:可以实现 子组件 与 父组件数据 的 双向绑定

.sync修饰符原理: :属性名@update:属性名 的合写

案例-封装城市下拉组件
<template>
  <div>
    <select :value="value" @change="handleChange">
      <option value="101">北京</option>

      <option value="102">上海</option>

      <option value="103">武汉</option>

      <option value="104">广州</option>

      <option value="105">深圳</option>

    </select>

  </div>

</template>

<script>
export default {
  props:{
    value:{
      type:String
    }
  },
  methods: {
  handleChange (e) {
    this.$emit('update:value', e.target.value)
  }
}
}
</script>

<style>
</style>
<template>
  <div class="app">
    <BaseSelect :value.sync="selectId"></BaseSelect>
    
    <BaseSelect :value="selectId" @update:value="selectId = $event">
    </BaseSelect>
    {{ selectId }}
  </div>
</template>
<script>
import BaseSelect from './components/BaseSelect.vue'
export default {
  data() {
    return {
      selectId: '102',
    }
  },
  components: {
    BaseSelect,
  },
}
</script>

<style>
</style>

ref和$refs

作用:利用 ref 和 $refs 可以用于 获取 dom 元素, 或 组件实例

ref和$refs特点:查找范围 → 当前组件内 (更精确稳定)

语法:

案例-获取dom元素-渲染图表

<template>
  <div class="app">
    <BaseChart></BaseChart>
  </div>

</template>

<script>
import BaseChart from './components/BaseChart.vue'
export default {
  components:{
    BaseChart
  }
}
</script>

<style>
</style>
<template>
  <div class="base-chart-box" ref="baseChartBox">子组件</div>

</template>

<script>
// yarn add echarts 或者 npm i echarts
import * as echarts from 'echarts'

export default {
  mounted() {
    // 基于准备好的dom,初始化echarts实例
    // const myChart = echarts.init(document.querySelector('.base-chart-box'))
    const myChart = echarts.init(this.$refs.baseChartBox)
    // 绘制图表
    myChart.setOption({
            // 大标题
            title: {
              text: '消费账单列表',
              left: 'center'
            },
            // 提示框
            tooltip: {
              trigger: 'item'
            },
            // 图例
            legend: {
              orient: 'vertical',
              left: 'left'
            },
            // 数据项
            series: [
              {
                name: '消费账单',
                type: 'pie',
                radius: '50%', // 半径
                data: [
                  { value: 1048, name: '球鞋' },
                  { value: 735, name: '防晒霜' }
                ],
                emphasis: {
                  itemStyle: {
                    shadowBlur: 10,
                    shadowOffsetX: 0,
                    shadowColor: 'rgba(0, 0, 0, 0.5)'
                  }
                }
              }
            ]
          })
  },
}
</script>

<style scoped>
.base-chart-box {
  width: 400px;
  height: 300px;
  border: 3px solid #000;
  border-radius: 6px;
}
</style>

案例-获取组件实例-登录组件

<template>
  <div class="box">
    <form>
      账号:<input type="text" v-model="user.uname" />
      <br />
      密码:<input type="password" v-model="user.pwd" />
    </form>
  </div>
</template>

<script>
export default {
  data() {
    return {
      user: {
        uname: "admin",
        pwd: "123",
      },
    };
  },
  methods:{
    login(){
      alert(JSON.stringify(this.user))
    }
  }
};
</script>

<style scoped>
.box {
  width: 300px;
  height: 100px;
  border: 1px solid #000;
  border-radius: 10px;
  display: flex;
  justify-content: center;
}

.box form {
  height: 70px;
  align-self: center;
}

.box form input:last-child {
  margin-top: 10px;
}
</style>
<template>
  <div class="app">
    <UserLogin ></UserLogin>
    <div>
      <button >获取用户数据</button>
      <button >重置表单</button>
    </div>
  </div>

</template>

<script>
import UserLogin from './components/UserLogin.vue'
export default {
  components:{
    UserLogin
  },
  methods:{
    reset(){
    }
  }
}
</script>

<style>
.base-chart-box {
  width: 400px;
  height: 300px;
  border: 3px solid #000;
  border-radius: 6px;
}
</style>

Vue异步更新 & $nextTick

Vue的异步更新特性

<template>
  <div class="app">
    <span ref="span">{{ num }}</span>
    <button @click="addone">+1</button>
  </div>
</template>

<script>
export default {
  data() {
    return {
      num: 1,
    };
  },
  methods: {
    addone() {
      this.num++;
      this.num++;
      console.log(this.num);//打印3
      console.log(this.$refs.span.innerHTML)  //❌还是拿到上一次的值1
    },
  },
};
</script>

<style>
</style>

Vue 在更新 DOM 时是异步执行的

  • 只要侦听到数据变化,Vue 将开启一个队列,并缓冲在同一事件循环中发生的所有数据变更
  • 如果同一个 watcher 被多次触发,只会被推入到队列中一次。
  • 在下一个的事件循环中(nextTick),Vue 刷新队列并执行实际 (已去重的) 任务(先进先执行)

上面代码中,当num的值改变时,由于dom更新是异步的,所以通过

this.$refs.span.innerHTML拿到的结果是不准确的。

如何解决?使用 $nextTick

$nextTick

$nextTick:等 DOM 更新后, 才会触发执行此方法里的函数体

语法:

  1. 回调函数写法: this.$nextTick(()=>{ })
  2. Promise写法: this.$nextTick().then(res=>{ })

<template>
  <div class="app">
    <span ref="span">{{ num }}</span>
    <button @click="addone">+1</button>
  </div>
</template>

<script>
export default {
  data() {
    return {
      num: 1,
    };
  },
  methods: {
    addone() {
      this.num++;
      this.num++;
      console.log(this.num);//打印3
      // console.log(this.$refs.span.innerHTML)  //❌还是拿到上一次的值1
      this.$nextTick(()=>{ console.log(this.$refs.span.innerHTML)})  // ✔️打印3
      this.$nextTick().then(res=>{console.log(this.$refs.span.innerHTML)}) // ✔️打印3
    },
  },
};
</script>

<style>
.base-chart-box {
  width: 400px;
  height: 300px;
  border: 3px solid #000;
  border-radius: 6px;
}
</style>

案例-表格行内编辑

需求:点击编辑按钮,切换到编辑框后自动聚焦

<template>
  <div class="app">
    <div>
      <input type="text" v-model="title" ref="inp" />
      <button>确认</button>
    </div>
    <div>
      <span>{{ title }}</span>
      <button @click="editFn">编辑</button>
    </div>
  </div>
</template>

<script>
export default {
  data() {
    return {
      title: "标题",
      isShowEdit: false
    };
  },
  methods: {
    
  },
};
</script>

<style>
</style>
<template>
  <div class="app">
    <div v-if="isShowEdit">
      <input type="text" v-model="title" ref="inp" />
      <button>确认</button>
    </div>
    <div v-else>
      <span>{{ title }}</span>
      <button @click="editFn">编辑</button>
    </div>
  </div>
</template>

<script>
export default {
  data() {
    return {
      title: "大标题",
      isShowEdit: false
    };
  },
  methods: {
    editFn() {
      // 1.显示文本框
      this.isShowEdit = true;
      // 2.让文本框聚焦 (会等dom更新完之后 立马执行nextTick中的回调函数)
      this.$nextTick(() => {
        console.log(this.$refs.inp);
        this.$refs.inp.focus();
      });

      // setTimeout(() => {
      //   this.$refs.inp.focus()
      // }, 0)
    },
  },
};
</script>

<style>
</style>

注意:$nextTick 内的函数体 一定是箭头函数,这样才能让函数内部的this指向Vue实例

回顾

  1. v-model原理 -> 用v-model对组件进行双向绑定
  2. :属性.sync -> 对组件进行双向绑定
v-model原理:父传子 + 子传父
用在组件上的规则:将 v-model 拆解成了 :value=""  @input=""
  子组件中我们就可以使用  props:['value']来接收传入的数据  , 使用 this.$emit('input',传的数据)
  来向父组件传递数据

:属性可以自定义.sync  -> 拆解 :属性=""   @update:属性=""
props:['属性']来接收传入的数据  , 使用 this.$emit('update:属性',传的数据)
  来向父组件传递数据
  1. 事件总线 Event Bus
  2. provide & inject
  3. ref & $refs
  4. $nextTick(等待Dom更新完毕后再出发) -> Vue Dom异步更新(事件循环)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2276458.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

mapbox基础,expressions表达式汇总

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言二、🍀Expressions简介2.1 expressions 操作符2.1.1 Data expressions2.1.2 Camera expressions2.2 Expressi…

一文清晰梳理Mysql 数据库

现在处于大四上学期的阶段&#xff0c;在大四下学期即将要进行毕业设计&#xff0c;所以在毕业设计开始之前呢&#xff0c;先将Mysql 数据库有关知识进行了一个梳理&#xff0c;以防选题需要使用到数据库。 1&#xff09;什么是数据库&#xff1f; 简单理解数据库&#xff0c…

基于大语言模型的组合优化

摘要&#xff1a;组合优化&#xff08;Combinatorial Optimization, CO&#xff09;对于提高工程应用的效率和性能至关重要。随着问题规模的增大和依赖关系的复杂化&#xff0c;找到最优解变得极具挑战性。在处理现实世界的工程问题时&#xff0c;基于纯数学推理的算法存在局限…

安装conda 环境

conda create -n my_unet5 python3.8 conda activate my_unet5

容器技术全面攻略:Docker的硬核玩法

文章背景 想象一下&#xff0c;一个项目终于要上线了&#xff0c;结果因为环境配置不一致&#xff0c;测试服务器一切正常&#xff0c;生产环境却宕机了。这是开发者噩梦的开始&#xff0c;也是Docker救世主角色的登场&#xff01;Docker的出现颠覆了传统环境配置的方式&#…

LabVIEW部署Web服务

目录 LabVIEW部署Web服务1、创建项目2、创建Web服务3、新建WebVI3.1、使用GET方法3.2、使用POST方法 4、 部署和对应URL4.1、应用程序&#xff1a;80804.2、本地调试&#xff1a;80094.3、NI Web服务器&#xff1a;9090(禁用) 5、测试5.1、测试GET方法5.2、测试POST方法 6、实际…

STM32 : 波特率发生器

波特率发生器 1. 发送器和接收器的波特率 波特率寄存器 (BRR): 在串行通信中&#xff0c;发送器和接收器的波特率是由波特率寄存器&#xff08;BRR&#xff09;中的一个值 DIV 来确定的。 2. 计算公式 计算公式: 详细解释 1. 波特率寄存器 (BRR) BRR: 波特率寄存器是一…

Excel数据叠加生成新DataFrame:操作指南与案例

目录 一、准备工作 二、读取Excel文件 三、数据叠加 四、处理重复数据&#xff08;可选&#xff09; 五、保存新DataFrame到Excel文件 六、案例演示 七、注意事项 八、总结 在日常数据处理工作中&#xff0c;我们经常需要将不同Excel文档中的数据整合到一个新的DataFra…

基于微信小程序的汽车销售系统的设计与实现springboot+论文源码调试讲解

第4章 系统设计 一个成功设计的系统在内容上必定是丰富的&#xff0c;在系统外观或系统功能上必定是对用户友好的。所以为了提升系统的价值&#xff0c;吸引更多的访问者访问系统&#xff0c;以及让来访用户可以花费更多时间停留在系统上&#xff0c;则表明该系统设计得比较专…

C#调用OpenCvSharp实现图像的开运算和闭运算

对图像同时进行腐蚀和膨胀操作&#xff0c;顺序不同则效果也不同。先腐蚀后膨胀为开运算&#xff0c;能够消除小斑点和细小的突出物、平滑图像以及改善边缘&#xff1b;先膨胀后腐蚀为闭运算&#xff0c;能够去除噪点、填补图像孔洞、连接邻近物体和平滑物体边界。   OpenCvS…

从 SQL 语句到数据库操作

1. SQL 语句分类 数据定义语言 DDL &#xff1a; 用于定义或修改数据库中的结构&#xff0c;如&#xff1a;创建、修改、删除数据库对象。create、drop alter 数据操作语言 DML &#xff1a; 用于添加、删除、更新数据库中的数据。select、insert alter、drop 数据控制语言 D…

django在线考试系统

Django在线考试系统是一种基于Django框架开发的在线考试平台&#xff0c;它提供了完整的在线考试解决方案。 一、系统概述 Django在线考试系统旨在为用户提供便捷、高效的在线考试环境&#xff0c;满足教育机构、企业、个人等不同场景下的考试需求。通过该系统&#xff0c;用…

【Spring Boot 应用开发】-04-01 自动配置-数据源-连接池

资源关闭 还记得上一节中的这段代码么&#xff1f; try {if (resultSet ! null) resultSet.close();if (preparedStatement ! null) preparedStatement.close();if (connection ! null) connection.close(); } catch (SQLException e) {e.printStackTrace(); }这是我们在查询…

AngularJs指令中出错:Error: $compile:nonassign Non-Assignable Expression

Expression {resumeId: item.resumeId} used with directive rwdsDelete is non-assignable! 在AngularJS中&#xff0c;$compile服务用于将指令编译成HTML。当你在模板中使用了一个表达式&#xff0c;但这个表达式不是一个左值&#xff08;即不能被赋值的表达式&#xff09;时…

Docker 的安装和基本使用[SpringBoot之Docker实战系列] - 第535篇

历史文章&#xff08;文章累计530&#xff09; 《国内最全的Spring Boot系列之一》 《国内最全的Spring Boot系列之二》 《国内最全的Spring Boot系列之三》 《国内最全的Spring Boot系列之四》 《国内最全的Spring Boot系列之五》 《国内最全的Spring Boot系列之六》 《…

聚铭网络当选中关村华安关键信息基础设施安全保护联盟理事单位

近日&#xff0c;在北京隆重举行的中关村华安关键信息基础设施安全保护联盟&#xff08;以下简称“联盟”&#xff09;第一届第四次会员大会上&#xff0c;聚铭网络凭借其在网络安全领域的卓越贡献和创新实力&#xff0c;成功当选为联盟的理事单位。此次大会吸引了来自政府机关…

CES 2025|全面拥抱端侧AI,美格智能在CES发布系列创新成果

要点&#xff1a; ▶ 在AI机器人领域&#xff0c;以高算力AI模组助力发布“通天晓”人形机器人和2款全新微小型AI机器人 ▶ 在AI硬件领域&#xff0c;发布消费级AI智能体产品——AIMO&#xff0c;引领个人专属的大模型时代 ▶ 在5G通信领域&#xff0c;发布全新5GWiFi-7 CPE…

VScode 配置 C语言环境

遇到的问题集合 mingw官方下载网站&#xff08;https://sourceforge.net/projects/mingw-w64/files/&#xff09;更新之后&#xff0c;与网上大多数教程上写的界面不同了。 网上大多数教程让下载这个&#xff1a; 但是现在找不到这个文件。 写hello.c文件时&#xff0c;报错&…

013:深度学习之神经网络

本文为合集收录&#xff0c;欢迎查看合集/专栏链接进行全部合集的系统学习。 合集完整版请参考这里。 深度学习是机器学习中重要的一个学科分支&#xff0c;它的特点就在于需要构建多层且“深度”的神经网络。 人们在探索人工智能初期&#xff0c;就曾设想构建一个用数学方式…

STM32-Flash存储

目录 1.0 闪存模块组织 2.0 Flash基本结构 3.0 Flash解锁 4.0 指针访问存储器地址 5.0 程序存储器编程 6.0 选项字节 7.0 选项字节编程 8.0 选项字节擦除 9.0 电子签名 10.0 手册解读 定义&#xff1a; STM32F1系列的FLASH包含程序存储器、系统存储器和选项字节三个部…