班级网站的建设,自己做影视类网站,重庆市做网站的公司有哪些,网站建设百家号Vue 篇#xff08;持续更新中#xff09; 1、v-for 和 v-if 可以混合使用吗#xff1f;为什么#xff1f;2、v-for 中为什么加 key#xff1f;3、事件默认有个 event 参数#xff0c;它是什么#xff1f;怎么使用#xff1f;事件被绑定到哪里#xff1f;4、vue 父子组… Vue 篇持续更新中 1、v-for 和 v-if 可以混合使用吗为什么2、v-for 中为什么加 key3、事件默认有个 event 参数它是什么怎么使用事件被绑定到哪里4、vue 父子组件如何通讯5、父子组件声明周期调用顺序6、vue 双向绑定原理响应式原理7、$nextTick 是什么为什么优先解决微任务8、vuex 是什么为什么使用9、keep-alive 是什么怎么实现的生命周期首次加载是先执行 activated 还是 deactivated为什么10、vue2 和 vue3 的区别11、history 和 hash 路由实现原理区别是什么12、请说出常用的设计模式5 种以上并举例实际项目中的使用场景。13、$emit 和 $on 的本质14、虚拟 dom 是什么原理优缺点15、mixin 是什么优缺点原理vue3 用什么取代了16、自定义指令原理17、事件绑定原理18、$set 的原理19、Vue3 比 Vue2 有什么优势20、Vue3 声明周期21、Composition API 和 Options API22、如何理解 ref、toRef 和 toRefs23、为何 ref 需要 value 属性24、Vue3 升级了哪些重要的功能 1、v-for 和 v-if 可以混合使用吗为什么
不可以v-for计算优先级比v-if高首先会把虚拟节点渲染出来然后再进行v-if判断。降低渲染性能
2、v-for 中为什么加 key
如果不使用 keyVue 会使用一种最大限度减少动态元素并且尽可能的尝试就地修改/复用相同类型元素的算法。key 是为 Vue 中 vnode 的唯一标记通过这个 keydiff 算法可以更准确、更快速
更准确因为带 key 就不是就地复用了在 sameNode 函数 a.key b.key 对比中可以避免就地复用的情况。所以会更加准确。更快速利用 key 的唯一性生成 map 对象来获取对应节点比遍历方式更快
3、事件默认有个 event 参数它是什么怎么使用事件被绑定到哪里
当事件没有参数则默认有个 event 参数如果有自定义参数则需要使用$event 传过去。
event 的构造函数是 MouseEvent即是原生的 event 对象event 被挂载到当前元素下即 event.target
4、vue 父子组件如何通讯
props、$emit 父组件使用动态数据传递子组件使用props接收可以使用数组/对象数据结构对象结构可以定义类型和默认值。 子组件使用$emit 事件回传自定义事件进行组件通讯 自定义事件就是使用一个额外的 js 文件其中声明一个 Vue 实例即可 $on$emit$off参数分别是注册函数名真实函数 $on绑定自定义事件 $emit调用自定义事件 $off在组件销毁时要及时销毁自定义事件否则可能会造成内存泄漏。在 beforeDestroy 中调用$off$refs 获取当前组件实例$parent$children 获取当前组件的父组件和子组件vuex Vue 中全局状态管理系统用于多个组件中数据共享。provideinject 上层组件提供下层组件注入使用。适用于组件库编写
5、父子组件声明周期调用顺序
加载渲染过程 父beforeCreate - 父created - 父beforeMount - 子beforeCreate - 子created - 子beforeMount - 子mounted - 父mounted更新过程 父beforeUpdate - 子beforeUpdate - 子updated - 父updated销毁过程 父beforeDestroy - 子beforeDestroy - 子destroyed - 父destroyed全流程 beforeCreate - created - beforeMount - mounted - beforeUpdate - updated - beforeDestroy - destroyed
6、vue 双向绑定原理响应式原理
采用数据劫持结合观察者模式的方式 通过 Object.defineProperty()来劫持各个属性只会劫持已经存在的属性的 settergetterdep 和 Watcher 实现依赖收集和派发更新的过程
vue 将 data 初始化为一个 Observer 并对对象中的每个值重写了其中的 get、setdata 中的每个 key都有一个独立的 dep依赖收集器。在 get 中向 dep依赖收集器添加了监听在 mount 时实例了一个 Watcher将收集器的目标指向了当前 Watcher在 data 值发生变更时触发 set触发了 dep依赖收集器中的所有监听的更新来触发 Watcher.update
7、$nextTick 是什么为什么优先解决微任务
nextTick 中的回调是在下次 DOM 更新循环结束之后执行的延迟回调。在修改数据之后立即使用这个方法获取更新后的 DOM。主要思路就是采用微任务优先的方式调用异步方法去执行 nextTick 包装的方法
8、vuex 是什么为什么使用
vuex 是专门为 vue 提供的全局状态管理系统用于多个组件中数据共享、数据缓存等。无法持久化、内部核心原理是通过创造一个全局实例 new Vue 主要包括以下几个模块
State定义了应用状态的数据结构可以在这里设置默认的初始状态。Getter允许组件从 Store 中获取数据mapGetters 辅助函数仅仅是将 store 中的 getter 映射到局部计算属性。Mutation是唯一更改 store 中状态的方法且必须是同步函数。Action用于提交 mutation而不是直接变更状态可以包含任意异步操作。Module允许将单一的 Store 拆分为多个 store 且同时保存在单一的状态树中。
9、keep-alive 是什么怎么实现的生命周期首次加载是先执行 activated 还是 deactivated为什么
作用与用法
开发中缓存组件使用 keep-alive 组件keep-alive 是 vue 内置组件keep-alive 包裹动态组件 component 时会缓存不活动的组件实例而不是销毁它们这样在组件切换过程中将状态保留在内存中防止重复渲染 DOM。
使用细节
结合属性 include 和 exclude 可以明确指定缓存哪些组件或排除缓存指定组件。vue3 中结合 vue-router 时变化较大之前是 keep-alive 包裹 router-view现在需要反过来用 router-view 包裹 keep-alive。
keep-alive 的中缓存的时候还运用了 LRU(Least Recently Used) 算法。
LRU(最近最少使用) 缓存机制
如果关键字 key 存在于缓存中则返回关键字的值否则返回 -1。如果关键字已经存在则变更其数据值如果关键字不存在则插入该组「关键字-值」。当缓存容量达到上限时它应该在写入新数据之前删除最久未使用的数据值从而为新的数据值留出空间
组件缓存后更新解决方案可以有以下两种
beforeRouteEnter在有 vue-router 的项目每次进入路由的时候都会执行 beforeRouteEnter。activated在 keep-alive 缓存的组件被激活的时候都会执行 activated 钩子。
原理
keep-alive 是一个通用组件它内部定义了一个 map缓存创建过的组件实例它返回的渲染函数内部会查找内嵌的 component 组件对应组件的 vnode如果该组件在 map 中存在就直接返回它。由于 component 的 is 属性是个响应式数据因此只要它变化keep-alive 的 render 函数就会重新执行。
参数
keep-alive 接收三个参数
include可传字符串、正则表达式、数组名称匹配成功的组件会被缓存exclude可传字符串、正则表达式、数组名称匹配成功的组件不会被缓存max可传数字限制缓存组件的最大数量超过 max 则按照 LRU 算法进行置换
include 和 exclude传数组情况居多。
生命周期
生命周期有activated 激活、deactivated 离开
activated 页面第一次进入的时候钩子触发的顺序是 created - mounted - activateddeactivated: 页面退出的时候会触发 deactivated
当再次前进或者后退的时候只触发 activated。
使用 keep-alive 会将数据保留在内存中如果要在每次进入页面的时候获取最新的数据需要在 activated 阶段获取数据承担原来 created 钩子中获取数据的任务。那么我们一般会在动态组件、路由组件去用到 keep-alive 组件。
10、vue2 和 vue3 的区别
响应式系统vue2 是 Object.defineProperty()vue3 使用的是 proxyvue3 全部由 ts 重构对 ts 支持更友好自定义渲染器composition APIvue3 可以存在多个根节点vue2 只能有一个
11、history 和 hash 路由实现原理区别是什么
hash 模式
1.location.hash 的值实际就是 URL 中 # 后面的东西 它的特点在于hash 虽然出现 URL 中但不会被包含在 HTTP 请求中对后端完全没有影响因此改变 hash 不会重新加载页面。2.可以为 hash 的改变添加监听事件
window.addEventListener(hashchange, funcRef, false);每一次改变 hashwindow.location.hash都会在浏览器的访问历史中增加一个记录。 利用 hash 的以上特点就可以来实现前端路由 “更新视图但不重新请求页面” 的功能了 特点兼容性好但是不美观 history 模式
利用了 HTML5 History Interface 中新增的 pushState() 和 replaceState() 方法。
这两个方法应用于浏览器的历史记录站在当前已有的 back、forward、go 的基础之上它们提供了对历史记录进行修改的功能。这两个方法有个共同的特点当调用他们修改浏览器历史记录栈后虽然当前 URL 改变了但浏览器不会刷新页面这就为单页应用前端路由 “更新视图但不重新请求页面” 提供了基础。 特点虽然美观但是刷新会出现 404 需要后端进行配置 12、请说出常用的设计模式5 种以上并举例实际项目中的使用场景。 1.工厂模式 - 传入参数即可创建实例 虚拟 DOM 根据参数的不同返回基础标签的 Vnode 和组件 Vnode 2.单例模式 - 整个程序有且仅有一个实例 vuex 和 vue-router 的插件注册方法 install 判断如果系统存在实例就直接返回掉 3.观察者模式 (响应式数据原理) 4.策略模式 策略模式指对象有某个行为,但是在不同的场景中,该行为有不同的实现方案-比如选项的合并策略 5.策略模式优化 if else 冗余代码 6.代理模式mini-vue proxy
13、$emit 和 $on 的本质
1.$on、$emit 基于发布订阅模式2.$on 用来收集所有的事件依赖他会将传入的参数 event 和 fn 作为 key 和 value 的形式存到 vm._events 这个事件集合里就像这样 vm._events[event][fn];3.而 $emit 是用来触发事件的他会根据传入的 event 在 vm_events 中找到对应的事件并执行 invokeWithErrorHandling(cbs[i], vm, args, vm, info)4.最后我们看 invokeWithErrorHandling 方法可以发现他是通过 handler.apply(context, args) 和 handler.call(context) 的形式执行对应的方法
14、虚拟 dom 是什么原理优缺点
用 js 模拟一颗 dom 树,放在浏览器内存中.当你要变更时,虚拟 dom 使用 diff 算法进行新旧虚拟 dom 的比较,将变更放到变更队列中,反应到实际的 dom 树,减少了 dom 操作。 虚拟 DOM 将 DOM 树转换成一个 JS 对象树,diff 算法逐层比较,删除,添加操作,但是,如果有多个相同的元素,可能会浪费性能,所以,react 和 vue-for 引入 key 值进行区分。
优点
保证性能下限 框架的虚拟 DOM 需要适配任何上层 API 可能产生的操作它的一些 DOM 操作的实现必须是普适的所以它的性能并不是最优的但是比起粗暴的 DOM 操作性能要好很多因此框架的虚拟 DOM 至少可以保证在你不需要手动优化的情况下依然可以提供还不错的性能即保证性能的下限无需手动操作 DOM 我们不再需要手动去操作 DOM只需要写好 View-Model 的代码逻辑框架会根据虚拟 DOM 和 数据双向绑定帮我们以可预期的方式更新视图极大提高我们的开发效率跨平台 虚拟 DOM 本质上是 JavaScript 对象,而 DOM 与平台强相关相比之下虚拟 DOM 可以进行更方便地跨平台操作例如服务器渲染、weex 开发等。
缺点:
无法进行极致优化 虽然虚拟 DOM 合理的优化足以应对绝大部分应用的性能需求但在一些性能要求极高的应用中虚拟 DOM 无法进行针对性的极致优化。 首次渲染大量 DOM 时由于多了一层虚拟 DOM 的计算会比 innerHTML 插入慢。
15、mixin 是什么优缺点原理vue3 用什么取代了
组件和组件之间有时候会存在相同的代码逻辑分为局部混入和全局混入我们希望对相同的代码逻辑进行抽取
Mixin 提供了一种非常灵活的方式来分发 Vue 组件中的可复用功能一个 Mixin 对象可以包含任何组件选项 — 其本质就是一个对象当组件使用 Mixin 对象时所有 Mixin 对象的选项将被 混入该组件本身的选项中
缺点
变量来源不明确不利于阅读多 mixin 可能会造成命名冲突mixin 和组件可能出现多对多的关系复杂度较高
Vue3 使用 Composition API 替代了优点
代码提取代码重用命名冲突解决
16、自定义指令原理
Vue 自定义指令有全局注册和局部注册两种方式。先来看看注册全局指令的方式通过 Vue.directive(id, [definition]) 方式注册全局指令。然后在入口文件中进行 Vue.use() 调用。 它的作用价值在于当开发人员在某些场景下需要对普通 DOM 元素进行操作。提高代码复用。 指令本质上是装饰器是 vue 对 HTML 元素的扩展给 HTML 元素增加自定义功能。vue 编译 DOM 时会找到指令对象执行指令的相关方法。 自定义指令有五个生命周期也叫钩子函数分别是 bind、inserted、update、componentUpdated、unbind
1.bind只调用一次指令第一次绑定到元素时调用。在这里可以进行一次性的初始化设置。2.inserted被绑定元素插入父节点时调用 (仅保证父节点存在但不一定已被插入文档中)。3.update被绑定于元素所在的模板更新时调用而无论绑定值是否变化。通过比较更新前后的绑定值可以忽略不必要的模板更新。4.componentUpdated被绑定元素所在模板完成一次更新周期时调用。5.unbind只调用一次指令与元素解绑时调用。
原理
1.在生成 ast 语法树时遇到指令会给当前元素添加 directives 属性2.通过 genDirectives 生成指令代码3.在 patch 前将指令的钩子提取到 cbs 中,在 patch 过程中调用对应的钩子4.当执行指令对应钩子函数时调用对应指令定义的方法
17、事件绑定原理
$on、$emit 是基于发布订阅模式的维护一个事件中心on 的时候将事件按名称存在事件中心里称之为订阅者然后 emit 将对应的事件进行发布去执行事件中心里的对应的监听器
18、$set 的原理
因为响应式数据 我们给对象和数组本身都增加了 __ob__ 属性代表的是 Observer 实例。当给对象新增不存在的属性 首先会把新的属性进行响应式跟踪 然后会触发对象 __ob__ 的 dep 收集到的 watcher 去更新当修改数组索引时我们调用数组本身的 splice 方法去更新数组
19、Vue3 比 Vue2 有什么优势
1.性能更好2.体积更小3.更好的 TS 支持4.更好的代码组织5.更好的逻辑抽离6.更多新功能
20、Vue3 声明周期
Options API 1.beforeDestroy 改为 beforeUnmount2.destroyed 改为 unmounted3.其他沿用 Vue2 的生命周期 Composition API setup 相当于整合了 beforeCreate 和 created 其余生命周期分别是写在 setup 中的函数 1.onBeforeMount()2.onMounted()3.onBeforeUpdate()4.onUpdated()5.onBeforeUnmount()6.onUnmounted()
21、Composition API 和 Options API
1.更好的代码组织2.更好的逻辑复用3.更好的类型推导
如何选择
1.不建议共用会引起混乱2.小型项目、业务逻辑简单用 options API3.中大型项目、逻辑复杂用 composition API
22、如何理解 ref、toRef 和 toRefs
ref
1.生成值类型的响应式数据2.可用于模版和 reactive3.通过.value 修改值
toRef
1.针对一个响应式对象reactive 封装的 prop2.创建一个 ref具有响应式3.两者保持引用关系
toRefs
1.将响应式对象reactive 封装转换为普通对象2.对象的每个 prop 都是对应的 ref不然 reactive 响应式直接解构会失去响应式3.两者保持引用关系
23、为何 ref 需要 value 属性
1.ref 是一个对象不丢失响应式值类型不能用 proxy 代理value 储存值2.通过 .value 属性的 get 和 set 实现响应式3.用于模版、reactive 时不需要 .value其他情况都需要
24、Vue3 升级了哪些重要的功能
1.createApp2.emits 属性在子组件中声明 emits options 父组件的绑定事件3.生命周期使用 setup 整合 beforeCreate 和 createdbeforeDestroy 改为 beforeUnmountdestroyed 改为 unmounted其他与 vue2 一致4.多事件处理在点击事件中写入多个处理函数用逗号分割5.Fragment可以存放多个根节点6.移除 .sync7.异步组件的写法需要从 Vue 引入 defineAsyncComponent使用这个函数包裹 import() 引入异步组件8.移除 filter双括号中用 | 分割转换含义9.Teleport主要场景就是把组件的嵌套层级提高10.Suspense用来加载异步组件未成功时的一些 loading主要实现原理就是具名插槽11.Composition API