| 内容 | 参考链接 |
|---|---|
| JavaScript 面试高频考点 | HTML、CSS、JavaScript、ES6、AJAX、HTTP 面试考点 |
| Vue2.x 面试高频考点 | Vue2.x 面试高频考点 |
| Vue3.x新增API | 生命周期,ref、toRef 和 toRefs 的理解和最佳使用方式 |
| Vue3.x升级的重要功能 | emits属性、生命周期、多事件、Fragment、移出.async、异步组件写法、移出 filter、Teleport、Suspense… |
| Vue3.x响应式 | Composition API的逻辑复用、Proxy实现响应式 |
| Vue3.x深入理解 | v-model参数用法、watch和watchEffect区别等 |
useMousePosition.js 文件
import { ref, onMounted, onUnmounted} from 'vue'
function useMousePosition() {
// 初始化坐标
const x = ref(0)
const y = ref(0)
// 更新坐标
function update(e) {
x.value = e.pageX
y.value = e.pageY
}
// 挂载:添加鼠标移动事件
onMounted(() => {
console.log('useMousePosition mounted');
window.addEventListener('mousemove', update)
})
// 销毁:删除鼠标移动事件
onUnmounted(() => {
console.log('useMousePosition unMounted');
window.removeEventListener('mousemove', update)
})
return {
x,
y
}
}
export default useMousePosition
App.vue 父组件
<template>
<MousePosition v-if="flag" />
<button @click="changeFlagHandler">change flag</button>
</template>
<script>
import MousePosition from "./components/index.vue";
export default {
data() {
return {
flag: true,
};
},
methods: {
// 实现组件的创建/销毁
changeFlagHandler() {
this.flag = !this.flag;
},
},
components: { MousePosition },
};
</script>
index.vue 子组件
<template>
<p>mouse position {{ x }} {{ y }}</p>
</template>
<script>
import useMousePosition from "./useMousePosition";
export default {
name: "MousePosition",
setup() {
// 解构 x 和 y
const { x, y } = useMousePosition();
return {
x,
y,
};
},
};
</script>
CompositionAPI复用
示例:对象通过 Proxy 实现响应式测试
const data = {
name: '杂货铺',
age: 20
}
const proxyData = new Proxy(data, {
// 监听获取
get(target, key, receiver) {
const result = Reflect.get(target, key, receiver)
console.log('get', key);
return result // 返回结果
},
// 监听设置
set(target, key, val, receiver) {
const result = Reflect.set(target, key, val, receiver)
console.log('set', key, val)
console.log('result', result); // true
return result // 是否设置成功
},
// 监听删除
deleteProperty(target, key) {
const result = Reflect.deleteProperty(target, key)
console.log('delete property', key);
console.log('result', result); // true
return result // 是否删除成功
}
})

示例:数组通过 Proxy 实现响应式测试
const data = ['a', 'b', 'c']
const proxyData = new Proxy(data, {
get(target, key, receiver) {
// 只处理本身(非原型的)属性
const ownKeys = Reflect.ownKeys(target) // 获取对象的键
if (ownKeys.includes(key)) {
console.log('get', key); // 监听
}
const result = Reflect.get(target, key, receiver)
console.log('get', key);
return result // 返回结果
},
set(target, key, val, receiver) {
// 重复的数据,不处理
const oldVal = target[key]
if(val === oldVal) {
return true
}
const result = Reflect.set(target, key, val, receiver)
console.log('set', key, val)
console.log('result', result); // true
return result // 是否设置成功
},
deleteProperty(target, key) {
const result = Reflect.deleteProperty(target, key)
console.log('delete property', key);
console.log('result', result); // true
return result // 是否删除成功
}
})


示例:使用 Proxy 实现响应式
// 创建响应式
function reactive(target = {}) {
if (typeof target !== 'object' || target == null) {
// 不是对象或数组
return target
}
// 代理配置
const proxyConf = {
get(target, key, receiver) {
// 只处理本身(非原型的)属性
const ownKeys = Reflect.ownKeys(target)
if (ownKeys.includes(key)) {
console.log('get', key); // 监听
}
const result = Reflect.get(target, key, receiver)
// 深度监听
// 性能如何提升的? 什么时候 get 到,什么时候去做响应式
return reactive(result) // 返回结果
},
set(target, key, val, receiver) {
// 重复的数据,不处理
const oldVal = target[key]
if (val === oldVal) {
return true
}
// 监听是已有的键还是新增的键
const ownKeys = Reflect.ownKeys(target)
if (ownKeys.includes(key)) {
console.log('已有的 key', key);
} else {
console.log('新增的 key', key);
}
const result = Reflect.set(target, key, val, receiver)
console.log('set', key, val)
return result // 是否设置成功
},
deleteProperty(target, key) {
const result = Reflect.deleteProperty(target, key)
console.log('delete property', key);
console.log('result', result); // tru return result // 是否删除成功
}
}
// 生成代理对象
const observed = new Proxy(target, proxyConf)
return observed
}
// 测试数据
const data = {
name: '杂货铺',
age: 21,
info: {
city: 'beijing'
}
}
const proxyData = reactive(data)
不积跬步无以至千里 不积小流无以成江海
点个关注不迷路,持续更新中…