| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> JavaScript知识库 -> 前端常见面试题(详解)——持续更新中 -> 正文阅读 |
|
[JavaScript知识库]前端常见面试题(详解)——持续更新中 |
目录 13.vue2中v-model是一个语法糖,那具体是怎么实现的? vue1.vue生命周期Vue 实例从创建到销毁的过程,就是生命周期。也就是从开始创建、初始化数据、编译模板、挂载Dom→渲染、更新→渲染、卸载等一系列过程 初始化?? 在beforeCreate生命周期函数执行的时候,data和method 还没有初始化 在created 生命周期函数执行的时候,data和method已经初始化完成 挂载 在beforeMount 生命周期函数执行的时候,已经编译好了模版字符串、但还没有真正渲染到页面中去 在mounted 生命周期函数执行的时候,已经渲染完,在视图中可以看到页面 更新 在beforeUpdate生命周期函数执行的时候,已经可以拿到最新的数据,但还没渲染到视图中去。 在updated生命周期函数执行的时候,已经把更新后的数据渲染到视图中去了。 销毁 在beforeDestroy 生命周期函数执行的时候,实例进入准备销毁的阶段、此时data 、methods 、指令等还是可用状态? 在destroyed生命周期函数执行的时候,实例已经完成销毁、此时data 、methods 、指令等都不可用? 三个不常用 keep-alive 主要用于保留组件状态或避免重新渲染。 activated只有在keep-alive 组件激活时调用。 deactivated只有在keep-alive 组件停用时调用。 errorCapured 当捕获一个来自子孙组件的错误时被调用。此钩子会收到三个参数:错误对象、发生错误的组件实例以及一个包含错误来源信息的字符串。此钩子可以返回 2.vue.use()在 install 里我们可以拿到 Vue 那么和 Vue 相关的周边工作都可以考虑放在
通过全局方法?
常见的注册场景
echarts 用 Vue.use() 来注册 main.js
echarts.js
3.自定义指令可以看这个??8个非常实用的vue自定义指令 除了默认的内置指令 ( 指令定义函数提供了几个钩子函数(可选):
接下来我们来看一下钩子函数的参数 (包括
除了 添加一个自定义指令,有两种方式:
创建全局指令: 需要传入指令名称以及一个包含指令钩子函数的对象,该对象的键即钩子函数的函数名,值即函数体,钩子函数可以有多个。
?局部自定义指令
应用场景 在做过的一个项目中,有一个上传头像的部分,上传了一张图片后,它的地址有错误,就需要自己给定一个默认的图片,就用到了自定义指令,v-images 当地址是错误的时候,就需要用到自己自定义的一个默认的图片 批量注册指令,新建?
?在?
4.computed和watch有何区别?计算属性computed : 1.
4.如果computed属性属性值是函数,那么默认会走get方法;函数的返回值就是属性的属性值;在computed中的属性都有一个get和一个set方法,当数据变化时,调用set方法。 5.使用? 给计算属性的变量赋值时使用全写? ? ? 需要进行计算求和的时候使用到
侦听属性watch: 1. 2.不支持缓存,数据变,直接会触发相应的操作
deep: 深度监听,为了发现对象内部值的变化,复杂类型的数据时使用 5.使用: 当进行全选与饭选的时候用到 监听一个基本数据时:
?监听一个引用数据类型时:
总结: 当目的是进?数值计算,且依赖于其他数据,那么推荐使用 当需要在某个数据发生变化的, 同时做?些稍复杂的逻辑操作,那么推荐使? 5.过滤器的使用filters过滤器就是一个函数, 传入值返回处理后的值,被用于转换格式,常见的文本格式化,将值转换为另一种形式 Vue3中舍弃了filters 过滤器,用函数替代了过滤器 过滤器可以用在两个地方:双花括号插值和v-bind表达式 全局的用Vue.filter():Vue.filter("过滤器名", (值) => {return "返回处理后的值"}) 局部的用filters属性:?filters: {过滤器名字: (值) => {return "返回处理后的值"} 过滤器传参: vue 变量 | 过滤器传参 多个过滤器: vue变量 | 过滤器1 | 过滤器2 应用场景: 时间格式化用到过滤器,有一个禁用和启用的状态改变用到了过滤器,人员员工的聘用形式用到了过滤器:过滤器的名字:function() {return}? 导入,全局过滤器注册, 使用{vue变量:过滤器名字} 6.vuex 的理解vuex的本质是一个对象,是响应式的,管理公共数据的工具,用于多个组件之间的数据共享。vuex的出现是为了解决web组件化开发的过程中,各组件之间传值的复杂和混乱的问题。 ?整个虚线部分就是Vuex,我们可以把它看成一个公共仓库store。store中有Actions(行为)、Mutations(变动)和State(状态)。整个的逻辑是组件通过Dispatch调用Actions中的方法,Actions通过Commit调用Mutations中的方法,Mutatisons改变State中的值。 vuex是全局状态管理库,可以通过它来进行全局数据流的管理。 state: 存放多个组件都用得到的公共数据,定义了应用状态的数据结构,可以在这里设置默认的初始状态(相当于data) mutations: 存放操作数据的方法,修改state (相当于methods) actions: 存放一些异步操作 (也可以进行一些同步处理) 注意: actions是不能直接修改state数据的, 需要提交mutation getters: 存放基于state计算出来的一些值 (相当于计算属性) modules: 分模块, 项目大了, 推荐分模块管理,用到映射,先从vuex中导入mapState,然后使用 注意点: 分模块了, 默认muations, actions, getters 注册到全局的, 一般会开启命名空间,语法: namespaced: true??可以解决不同模块命名冲突的问题,将不同模块的namespaced:true,之后在不同页面中引入getter、actions、mutations时,需要加上所属的模块名。 简述vuex的数据传递流程 当组件进行数据修改的时候,我们需要调用dispatch来触发actions里面的方法。actions里面的每个方法中都会有一个 commit 方法,当方法执行的时候,会通过commit 来触发 mutations 里面的方法来进行数据的修改,?mutations 里面的每个函数都会有一个 state 参数, 这样就可以在 mutations 里面进行state 的数据修改, 当数据修改完成后, 会传导给页面,页面的数据也会发生改变。 7.父子组件的生命周期钩子0. 加载渲染过程 父 beforeCreate 父 created 父 beforeMount 子 beforeCreate 子 created 子 beforeMount 子 mounted 父 mounted 1. 子组件更新过程 子 beforeUpdate 子 updated 2. 父组件更新过程? 父组件更新的数据子组件有使用! 父 beforeUpdate 子 beforeUpdate 子 updated 父 updated 3. 销毁过程 父 beforeDestroy 子 beforeDestroy 子 destroyed 父 destroyed 8.v-mode和.sync的对比v-model 只能使用一次,一个页面中只能使用一次 .async可以使用多次 vue3中没有async 用函数来替代 v-model = " 父组件属性"? ? ?就相当于?? :value= “父组件的属性” @ input = "父组件的属性" = $event
9.对nextTick的理解vue是响应式的框架:状态的变化能自动更新视图。不过,这个过程是异步的。nextTick就是用来实现这个异步过程的核心函数。 在updated生命周期钩子函数里面可以访问到更新后的DOM,或者在this.$nextTick里的函数体中访问到 有两种情况会调用nextTick:
那么nextTick内部做了什么呢? 主要是两件事:
应用场景1: 父组件的状态通过props传递给子组件使用时,当父组件的状态改动后,想通过子组件的实例获取内部的props, 伪代码如下:
?应用场景2:? ?静默刷新 ,无感知的刷新 对象的新增属性vue无法响应式的更新,原有属性可正常更新 数组的能修改源数组的方法能实现数组的响应式更新,但是如果直接修改数组的值无法响应式更新 vue 不能检测对象属性的添加与删除? ? 通过这个方法 this.$set(对象,属性名,修改后的值) 可以解决? 但是有时候项目中的属性多了,不知道改哪一个,就可以使用 静默刷新 解决
应用场景3 关于vue 引入 echarts获取不到DOM节点的问题 vue3中需要这样引入并把echarts挂载到vue原型上 遇到的问题
10.哈希路由和history路由的区别单页面(SPA)应用是在移动互联时代诞生的,它的目标是不刷新浏览器,而通过感知地址栏中的变化来决定内容区域显示什么内容。要达成这个目标,我们要用到前端路由技术,具体来说有两种方式来实现:hash模式和history模式。 一是原理不同。hash模式的实现原理是通过监听hashChange事件来实现的。history模式是通过调用 history.pushState方法(或者replaceState) 并且 监听popstate事件来实现的。history.pushState会追加历史记录,并更换地址栏地址信息,但是页面不会刷新,需要手动调用地址变化之后的处理函数,并在处理函数内部决定跳转逻辑;监听popstate事件是为了响应浏览器的前进后退功能。 二是表现不同。hash模式会在地址栏中有#号,而history模式没有;同时由于history模式的实现原理用到H5的新特性,所以它对浏览器的兼容性有要求(IE >= 10)。 三是history模式开发的SPA项目,需要服务器端做额外的配置,否则会出现刷新白屏(链接分享失效)。原因是页面刷新时,浏览器会向服务器真的发出对这个地址的请求,而这个文件资源又不存在,所以就报404。处理方式就由后端做一个保底映射:所有的请求全部拦截到index.html上。 11. 组件之间的传参方式props $emit eventBus ref parents children vuex v-model .sync provide inject
12.mutation和action的使用区别action 和 mutations 也很类似,主要的区别在于mutations 只能是同步操作,action 可以包含异步操作,而且可以通过 action 来提交 mutations mutations 有一个固有参数 state,接收的是 Vuex 中的 state 对象 action 也有一个固有参数 context,但是 context 是 state 的父级,包含 state、getters Vuex 的仓库是 store.js,将 axios 引入,并在 action 添加新的方法 分发调用action: this.$store.dispatch('action中的函数名',发送到action中的数据) 在组件中提交 Mutation: this.$store.commit(“mutation函数名”,发送到mutation中的数据) 在action中提交mutation :
13.vue2中v-model是一个语法糖,那具体是怎么实现的?v-model 原理: 从接触Vue我们就知道 v-model是实现数据双向绑定的 那他怎么实现绑定的原理的呢? 其实v-model本质上就是语法糖 在使用 v-model 后既绑定了数据 又添加了一个@input事件监听
当在input元素中使用v-model实现双数据绑定,其实就是在输入的时候触发元素的input事件,通过这个语法糖,也能够实现父子组件数据的双向绑定
$event 指代当前触发的事件对象。 $event.target 指代当前触发的事件对象的dom $event.target.value 就是当前dom的value值 在@input方法中,value => str? ? ? ? ? 在:value中:str => value 如此,形成了一个闭环,也就是所说的数据的双向绑定。 当我们在一个自定义组件上使用v-model并不能实现双向绑定,因为自定的组件并没有默认的value和input事件,在使用时,我们需要按照上面那样显式的去声明定义这些东西。这时,model选项就派上用场了,在定义组件的时候,指定prop的值和监听的事件。
14.v-if和v-show的区别v-show 和v-if都是true的时候显示,false的时候隐藏 但是:false的情况下 v-show是采用css样式的display:none 来控制显示与隐藏的 v-if是采用 DOM 元素的 创建与销毁 来 控制显示与隐藏的 如果需要频繁切换显示隐藏需要使用v-show 应用场景 自己做的项目中,有一个编辑的弹框的组件,在标签页点击修改弹框上的内容有所属学科与目录名称,但是从学科页面点击就只有目录名称,他们公用一个弹框组件,显示的内容不一样,用v-show 来解决的。 15.循环的时候,为什么一定要绑定key:key是给v-for循环生成标签颁发唯一标识的,key的作用主要是为了高效的更新虚拟DOM 用于性能的优化 如果没有唯一key , 页面上删除一条标签, 由于并不知道删除的是那一条! 所以要把全部虚拟dom重新渲染 如果知道key为x标签被删除掉, 只需要把渲染的dom为x的标签去掉即可! 虚拟DOM? ?本质就是一个对象,保存DOM的关键信息 虚拟DOM的好处,提高DOM更新的性能,不频繁操作真实DOM (在内存中找到变化的部分,再更新真实DOM? ——打补丁) 16.vue2 常用的指令有哪些v-for 根据数组的个数, 循环数组元素的同时还生成所在的标签 v-show 显示内容 v-if 显示与隐藏 v-else 必须和v-if连用 不能单独使用 否则报错 v-bind 动态绑定 作用: 及时对页面的数据进行更改, 可以简写成:分号 v-on 给标签绑定函数,可以缩写为@,例如绑定一个点击函数 函数必须写在methods里面 v-text 解析文本 v-html 解析html标签 v-slot? ?简写 # 17.vue2和vue3的区别vue3中没有$on, filter过滤器 .sync? ? ?生命周期? ? 选项api 与 组合api 1.为什么要有vue3 我们使用vue2常常会遇到一些体验不太好的地方,比如:
vue3的出现就是为了解决vue2的弊端,其composition API很好的解决了逻辑复用的问题,而且vue3源码就是用ts写的,对ts的支持非常好。我们在开发项目过程中可以使用ts的加持,使代码更加健壮。 2.vue3的优点
3.响应式原理的不同 Vue2.x实现双向数据绑定原理,是通过es5的 Object.defineProperty,根据具体的key去读取和修改。其中的setter方法来实现数据劫持的,getter实现数据的修改。但是必须先知道想要拦截和修改的key是什么,所以vue2对于新增的属性无能为力,比如无法监听属性的添加和删除、数组索引和长度的变更,vue2的解决方法是使用Vue.set(object, propertyName, value) 等方法向嵌套对象添加响应式。 Vue3.x使用了ES2015的更快的原生proxy 替代 Object.defineProperty。Proxy可以理解成,在对象之前架设一层“拦截”,外界对该对象的访问,都必须先通过这层拦截,因此提供了一种机制,可以对外界的访问进行过滤和改写。Proxy可以直接监听对象而非属性,并返回一个新对象,具有更好的响应式支持 4.生命周期的不同 beforeCreate -> 请使用 setup() created -> 请使用 setup() beforeMount -> onBeforeMount mounted -> onMounted beforeUpdate -> onBeforeUpdate updated -> onUpdated beforeDestroy -> onBeforeUnmount destroyed -> onUnmounted errorCaptured -> onErrorCaptured 如果要想在页面中使用生命周期函数,以往vue2的操作是直接在页面中写入生命周期,而vue3是需要去引用的,这就是为什么3能够将代码压缩到更低的原因 5.默认项目目录结构的不同 vue3移除了配置文件目录,config 和 build 文件夹,移除了 static 文件夹,新增 public 文件夹,并且 index.html 移动到 public 中,在 src 文件夹中新增了 views 文件夹,用于分类视图组件和公共组件 18.vue2中响应式的原理vue.js 是 采 用 数 据 劫 持 结 合 发 布 者 - 订 阅 者 模 式 的 方 式 , 通 过Object.defineProperty()来劫持各个属性的 setter,getter,在数据变动时发布消息给订阅者,触发相应的监听回调。 MVVM 作为数据绑定的入口,整合 Observer、Compile 和 Watcher 三者,通过 Observer 来监听自 己的 model 数据变化,通过Compile 来解析编译模板指令,最终利用 Watcher 搭起 Observer 和Compile 之间的 通信桥梁,达到数据变化 -> 视图更新;视图交互变化(input)数据 model 变更的双向绑定效果 JS1.js的数据类型 8种1、【数据类型分类】 -js中数据类型分为两大类:基本数据类型和引用数据类型 -基本数据类型分为:数值类型number、字符串string、布尔类型boolean、null、undefined -引用数据类型分为:对象类型object、函数function, 对象类型又分为:Object对象、Array数组、RegExp正则、Date时间对象、Math数学对象 ES6 中新增了一种 Symbol 。这种类型的对象永不相等,即始创建的时候传入相同的值,可以解决属性名冲突的问题,做为标记 2、【基本数据类型和引用数据类型的区别】 - 基本数据类型存储在栈内存中,引用数据类型存储在堆内存中 - 数据存储时,基本数据类型在变量中存的是值,引用数据类型在变量中存储的是空间地址 - 基本数据操作的是值,引用数据类型操作的是空间地址 2.如何判断数据类型 5种1.typeof(除了null 其他的都可以判断,判断原始数据)
2.instanceof (通过原型链可以正确判断对象数据类型) instanceof 可以判断一个对象是否是某个构造函数的实例 原理 因为A instanceof B 可以判断A是不是B的实例,返回一个布尔值,由构造类型判断出数据类型
3.通过Object下的toString.call()方法来判断 (最准确)
4.根据对象的contructor判断
5.jq中判断数据类型的方法
3.原型链每个对象都会在其内部初始化一个属性,就是 prototype(原型)对象,当我们访问一个对象的某个属性时,会先在这个对象本身属性上查找,如果没有找到,这个对象内部不存在这个属性,那么他就会去它的proto隐式原型即它的构造函数的prototype里查找这个属性,如果还没有找到就会再在构造函数的prototype的proto中查找,直到查到Object.prototype.proto为nul,,这样一层一层向上查找就会形成一个链式结构,我们称为原型链 应用:原型链是实现继承的主要方法 4.闭包闭包(closure)指有权访问另一个函数作用域中变量的函数。简单理解就是 ,一个作用域可以访问另外一个函数内部的局部变量。(闭包是一个受保护的变量空间) 在一个函数里面嵌套函数,里面的函数需要用到外面函数定义的变量,return出来为了延迟局部变量的生命周期 作用:延伸变量的作用范围,闭包函数 中的局部变量不会等着闭包函数执行完就销毁, 因为还有别的函数要调用它,只有等着所有的函都调用完了,它才会销毁,... ?应用场景 setTimeout 回调, 函数防抖 ..........很多 5.如何判断this的指向普通函数, 定时器函数, 立即执行函数中:this指向全局对象window 以函数形式调用时,this 永远都是window 以方法的形式调用时,this是调用方法的对象 事件处理函数中:this指向事件触发对象 以构造函数的形式调用时,this 是新创建的那个对象,构造函数的实例 使用call和apply调用时,this 是指定的那个对象 箭头函数:箭头函数的this 看外层是否有函数 如果有,外层函数的this 就是内部箭头函数的this 如果没有,就是window 特殊情况:通常意义上 this 指针指向为最后调用它的对象。这里需要注意的一点就是 如果返回值是一个对象,那么this 指向的就是那个返回的对象,如果返回值不是一个对象那么 this 还是指向函数的实例 this的指向在函数定义的时候是确定不了的,只有函数执行的时候才能确定this到底指向谁,一般情况下this的最终指向的是那个调用它的对象。谁调用这个 this 就指向谁 6.call和apply 和bind的使用共同点: 1、都是用来改变函数的this对象的指向的。 2、第一个参数都是 this要指向的对象。 3、都可以利用后续参数传参。 不同点: call方法调用一个函数, 其具有一个指定的this 值和分别地提供的参数(参数的列表)。 注意:该方法的作用和 apply() 方法类似,只有一个区别,就是 call()方法接受的是若干个 参数的列表,而apply()方法接受的是一个包含多个参数的数组 方法调用一个具有给定this 值的函数,以及作为一个数组(或类似数组对象)提供的参 数。 注意:call()方法的作用和 apply() 方法类似,区别就是call()方法接受的是参数列表,而 apply()方法接受的是一个参数数组 bind()方法创建一个新的函数,当这个新的函数被调用时,其this 值为提供的值,其参 数列表前几项,置为创建时指定的参数序列 call: 方法.call(对象,参数1,参数2,参数3…) call参数是一个一个挨着写的 apply:方法.apply(对象,[参数1,参数2,参数…]) apply是2个参数,第二个参数是一个大数组? bind()方法主要就是将函数绑定到某个对象,bind()会创建一个函数,函数体内的this对象的值会被绑定到传入bind()第一个参数的值,例如,f.bind(obj),实际上可以理解为obj.f(),这时,f函数体内的this自然指向的是obj 7.数组常用的api1.向数组添加元素的方法 Array.push() 在数组的末尾处添加;会改变原有数组,返回值是添加数据后数组的新长度。 Array.unshift() 在数组的开头处添加,会改变原有数组返回值是添加数组后数组的新长度 splice(index,0,value1,value2,...) 向数组的指定index处插入;返回的是被添加的元素;会改变原有数组 2.向数组删除元素的方法 pop() 从尾部删除一个元素;返回被删除的元素;会改变原有数组 shift() 从头部删除一个元素;返回被删除的元素;会改变原有数组 splice(index,howmany) 在index处删除howmany个元素,返回的是被删除掉的元素的集合;会改变原有数组 splice(从哪个位置开始删除, 删除几个元素) 3.数组排序 reverse() 数组的翻转 颠倒数组中元素的排序 无参数 会改变原来的数组 返回新的数组 sort() 数组的排序 冒泡排序
按照指定规则排序,改变原有数组 数组的sort()方法里面的参数就是一个回调函数,有俩个值,返回值如果是第一个值减去第二个值,那么就是从小到大排序,如果是第二个数减去第一个值就是从大到小排序。 4.数组链接 concat() 数组连接,返回的是新数组 join() 将数组的每个元素以指定分隔符(默认为‘,’)连接成字符, 返回该连接完成后的字符串 5.数组截取 slice(start,end) 从start起始索引处,截取到end结束索引处,返回截取到的元素集合 6.数组转换 toString() 转换为字符串,和不传参数的join()一致 7.数组查找 indexOf(val[,fromIndex=0]) 数组的indexOf方法用来查找数组中某个val值第一次出现的索引,找不到就返回-1 lastIndexOf(val[,fromIndex=arr.length=-1]) 数组的lastIndexOf()方法用来查找数组中的某个val值第一次出现的索引,找不到返回 -1,lastIndexOf是从数组的最后往前找 8.数组累加 reduce()方法接收一个函数作为累加器,reduce为数组中的每一个元素依次执行回调函数,接收四个参数:初始值(上一次回调返回的值),当前元素,当前索引,原数组。 语法:reduce(callback, [initialValue]
reduce()被一个非空数组调用(如果被非空数组调用返回undefined),接收两个参数,一个callback和一个设置的累加初始值,需要注意的是如果给reduce()传入了初始值,则在该值的基础是做累加操作,如果初始值不存在,则total为数组的第一项,currentValue为下一项,在第一项的基础上累加,相当于设置初始值为0,然后逐步累加。 9.?includes():搜索指定元素是否在数组中,若在,则返回true,否则返回false。第二个参数指定开始搜索的位置。 10.对数组做迭代遍历操作,并且返回相应的值,迭代的入参是回调函数 map():操作数组每个元素,最终返回一个新的数组 forEach():操作数组的每个参数,没有返回值 filter():操作数组的每个元素,每次操作的结果为true的元素插入最终返回的数组(过滤) some():判断当前数组中是否有符合条件的元素,没有返回false,,否则返回true every():判断当前数组中是否含有所有元素都符合条件,都符合则返回true,否则返回false 8.什么是回调地狱,如何解决层层嵌套的回调函数 由于回调函数是异步的,每一层的回调函数都需要依赖上一层的回调执行完,所以形成了 层层嵌套的关系最终形成了回调地狱。例如:定时器中再写定时器再写定时器,这种就形成了 通常所说的回调地狱 解决: 1、避免函数的嵌套 2、模块化开发 3、使用 Promise 解决 如果上一个 .then() 方法中返回了一个新的 Promise 实例对象,则可以通过下一个 .then() 继续进行处理。通 过 .then() 方法的链式调用,就解决了 9.promise的使用promise 是用来解决回调地狱的问题的 Promise的构造函数接收一个参数,是函数,并且传入两个参数:resolve,reject,分别表示异步操作执行成功后的回调函数和异步操作执行失败后的回调函数 三个状态: 等待, 执行resolve, 拒绝 reject? ? ? ? ?默认情况下是等待状态pending, 1、自己身上有 all、reject、resolve 这几个方法 2、原型上有 then、catch 等方法 3、一旦建立,就无法取消,这是它的缺点 10.new的背后做哪些事情创建一个对象。让this指向这个对象。设置原型链 空对象指向构造函数的原型对象。返回这个对象 1、创建一个空对象: 并且 this变量引入该对象,同时还继承了函数的原型 2、设置原型链 空对象指向构造函数的原型对象 3、执行函数体 修改构造函数this 指针指向空对象,并执行函数体 4、判断返回值 返回对象就用该对象,没有的话就创建一个对象 11.防抖和节流
防抖(debounce): input 输入框 一直打字,直到停止打字后,才会发请求调用防抖的函数 search搜索联想,用户在不断输入值时,用防抖来节约请求资源。 window 触发resize 的时候,不断的调整浏览器窗口大小会不断的触发这个事件,用防抖 来让其只触发一次 实现输入框的防抖
节流(throttle) 鼠标不断点击触发,mousedown(单位时间内只触发一次) 监听滚动事件,比如是否滑到底部自动加载更多,用throttle 来判断 使用节流优化鼠标跟随效果
12.高阶函数将函数作为参数, 返回函数 高阶函数是对其他函数进行操作的函数,操作可以是将它们作为参数,或者是返回它们。 简单来说,高阶函数是一个接收函数作为参数或将函数作为输出返回的函数。(回调函数)(闭包) 例如,Array.prototype.map,Array.prototype.filter 和 Array.prototype.reduce 是语言中内置的一些高阶函数。 13.函数柯里化是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术
?实际上就是把add函数的x,y两个参数变成了先用一个函数接收x然后返回一个函数去处理y参数。现在思路应该就比较清晰了,就是只传递给函数一部分参数来调用它,让它返回一个函数去处理剩下的参数
14.for in和for of的区别1、推荐在循环对象属性的时候使用for...in,在遍历数组的时候的时候使用 for...of 2、for...in循环出的是 key,for...of循环出的是value 3、注意,for...of是ES6新引入的特性。修复了ES5 引入的for...in的不足 4、for...of不能循环普通的对象,需要通过和Object.keys()搭配使用 ?15.递归及应用场景一个函数可以调用它本身 树形结构数据的转换** 深拷贝 一般树状结构的都可以使用递归查询,比如 查询地区,树状的菜单等等,递归比普通的算法耗内存,谨慎使用。还有一种叫作“尾递归”就是把上一个方法的返回值当作参数传给下一个方法,不用像递归再向上返回 16.eventloop事件循环什么是Event Loop JavaScript的事件分两种,宏任务(macro-task)和微任务(micro-task) 宏任务:包括整体代码script,setTimeout,setInterval 微任务:Promise.then(非new Promise),process.nextTick(node 中) 事件的执行顺序——先执行宏任务,然后执行微任务,任务有同步的任务和异步的任务, 同步的进入主线程,异步的进入 EventTable并注册函数,异步事件完成后,会将回调函数放在 队列中,如果还有异步的宏任务,那么就会进行循环执行上述的操作。 事件循环先执行宏任务,其中同步任务立即执行,异步任务加载到对应的 EventQueue 中, 微任务也加载到对应的微任务的EventQueue 中,所有的同步微任务执行完之后,如果发现微 任务的EventQueue 中有未执行完的任务,先执行他们这样算是完成了一轮事件循环。接下来 查看宏任务的队列中是否有异步代码,有的话执行第二轮的事件循环,以此类推。 17.深拷贝与浅拷贝浅拷贝 浅拷贝只复制指向某个对象的指针而不复制对象本身,新旧对象还是共享同一块内存。 Object.assign() ES6的三个点 展开符? ...? 深拷贝 但深拷贝会另外创造一个一模一样的对象,新对象跟原对象不共享内存,修改新对象不会改到原对象。 JSON.parse(JSON.stringify(obj)) 网络1.https和http的区别1、https协议需要到CA申请证书,一般免费证书较少,因而需要一定费用。 2、http是超文本传输协议,信息是明文传输,https则是具有安全性的ssl/tls加密传输协议。 3、http和https使用的是完全不同的连接方式,用的端口也不一样,前者是80,后者是443。 4、http的连接很简单,是无状态的;HTTPS协议是由SSL/TLS+HTTP协议构建的可进行加密传输、身份认证的网络协议,比http协议安全 2.常见http的状态码- 状态代码由三位数字组成,第一个数字定义了响应的类别,且有五种可能取值。 ? 200 OK:客户端请求成功。 3.如何减少重绘和回流什么是重绘Repaint 重排 (回流 reflow) 重绘:当元素的一部分属性发生改变,如外观、背景、颜色等不会引起布局变化,只需要浏览器 根据元素的新属性重新绘制,使元素呈现新的外观叫做重绘。 重排(回流):当render 树中的一部分或者全部因为大小边距等问题发生改变而需要DOM 树重 新计算的过程 重绘不一定需要重排(比如颜色的改变),重排必然导致重绘(比如改变网页位置) 方法: 需要要对元素进行复杂的操作时,可以先隐藏(display:"none"),操作完成后再显示 需要创建多个DOM节点时,使用DocumentFragment 创建完后一次性的加入document 缓存Layout属性值,如:var left = elem.offsetLeft; 这样,多次使用 left 只产生一次回流 尽量避免用table 布局(table元素一旦触发回流就会导致table 里所有的其它元素回流) 避免使用css 表达式(expression),因为每次调用都会重新计算值(包括加载页面) 尽量使用 css 属性简写,如:用 border 代替 border-width, border-style, border-color 4.描述输入url地址到网页展示的过程
5.跨域问题及解决方案1.通过 jsonp 跨域 2.跨域浏览器 3.开发环境中跨域的解决proxy:代理 原理就是:服务器与服务器之间没有跨域的问题,通过服务器进行中转, proxy进行代理 在vue.config.js文件中配置proxy -- p小写, 配置完要重启脚手架
4.CORS 核心思想 ,一般是后端用的 5.通过反向代理? 上线时候用的 6.get请求和post请求的区别
7.http的协议的三个内容通用头域、请求消息、响应消息和主体信息 包含请求方法、URI、HTTP 版本信息 8.请求头中的contentType有什么用处enctype 属性规定在发送到服务器之前应该如何对表单数据进行编码。 默认地,表单数据会编码为 "application/x-www-form-urlencoded"。就是说,在发送到服务器之前,所有字符都会进行编码(空格转换为 "+" 加号,特殊符号转换为 ASCII HEX 值)。 用来设置程序间传递信息进行编码 属性规定发送到服务器端之前,应该如何对表单数据进行编码,所有的字符都会进行编码,空格转为+ 特殊的符号转为ASCLL值 ?dom1.事件冒泡和事件捕获事件冒泡:从内到外? ?text -- div-- body-- document -- window 事件捕获:从外到内? window-- document -- body -- div -- text 2.?事件委托给父元素注册事件,利用事件冒泡,当子元素的事件触发,会冒泡到父元素,然后去控制相应的子元素 优点: 1、减少事件注册,节省内存。 2、在table 上代理所有td 的click事件。 3、在ul上代理所有 li的click 事件。 4、简化了dom节点更新时,相应事件的更新。 5、不用在新添加的li上绑定click事件。 6、当删除某个li时,不用移解绑上面的click事件。 缺点: 1、事件委托基于冒泡,对于不冒泡的事件不支持 2、层级过多,冒泡过程中,可能会被某层阻止掉。 3、理论上委托会导致浏览器频繁调用处理函数,虽然很可能不需要处理。所以建议就近委托, 比如在table上代理td,而不是在 document 上代理td。 4、把所有事件都用代理就可能会出现事件误判。比如,在 document 中代理了所有button 的 click事件,另外的人在引用改js 时,可能不知道,造成单击button 触发了两个click事件 ?3.如何添加和删除事件addEventListener()和removeEventListener() addEventListener()与removeEventListener()用于处理指定和删除事件处理程序操作。 它们都接受3个参数:如 addEventListener("事件名" , "事件处理函数" , "布尔值"); (注:事件名不含"on",如“click”) ? |
|
JavaScript知识库 最新文章 |
ES6的相关知识点 |
react 函数式组件 & react其他一些总结 |
Vue基础超详细 |
前端JS也可以连点成线(Vue中运用 AntVG6) |
Vue事件处理的基本使用 |
Vue后台项目的记录 (一) |
前后端分离vue跨域,devServer配置proxy代理 |
TypeScript |
初识vuex |
vue项目安装包指令收集 |
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 | -2025/1/8 23:56:20- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |