vue遇到的问题和解决方法,vue十大难点
方案有这么些:cnpm : 国内对npm的镜像版本/*cnpm website: https://npm.taobao.org/*/npm install -g cnpm --registry=https://registry.npm.taobao.org// cnpm 的大多命令跟 npm 的是一致的,比如安装,卸载这些yarn 和 npm 改源大法//使用 nrm 模块 : www.npmjs.com/package/nrmnpm confi
Q1:安装超时(install timeout)
方案有这么些:
cnpm : 国内对npm的镜像版本/*cnpm website: https://npm.taobao.org/*/npm install -g cnpm --registry=https://registry.npm.taobao.org// cnpm 的大多命令跟 npm 的是一致的,比如安装,卸载这些yarn 和 npm 改源大法//使用 nrm 模块 : www.npmjs.com/package/nrmnpm config : npm config set registry https://registry.npm.taobao.orgyarn config : yarn config set registry https://registry.npm.taobao.orgQ2:安装一些需要编译的包:提示没有安装python、build失败等
因为一些 npm 的包安装需要编译的环境,mac 和 linux 都还好,大多都齐全 window 用户依赖 visual studio 的一些库和python 2 , windows的小伙伴都装上: windows-build-tools python 2.x
Q3:can’t not find ‘xxModule’ - 找不到某些依赖或者模块
这种情况一般报错信息可以看到是哪个包抛出的信息.,一般卸载这个模块,安装重新安装下即可.
Q4:data functions should return an object
这个问题是 vue 实例内,单组件的data必须返回一个对象;如下
export default { name: 'page-router-view', data () { return { tabs: [ { title: '财务信息', url: '/userinfo' }, { title: '帐号信息', url: '/userinfo/base' } ] } }}为什么要 return 一个数据对象呢? 官方解释如下: data 必须声明为返回一个初始数据对象的函数,因为组件可能被用来创建多个实例。如果 data 仍然是一个纯粹的对象,则所有的实例将共享引用同一个数据对象!简言之,组件复用下,不会造成数据同时指向一处,造出牵一发而动全身的破问题...
Q5:我给组件内的原生控件添加事件,怎么不生效了!!!
{{item.menuName}} {{item.menuName}}Q6:我用了 axios , 为什么 IE 浏览器不识别(IE9 )
那是因为 IE 整个家族都不支持 promise, 解决方案:npm install es6-promise// 在 main.js 引入即可// ES6的polyfillrequire("es6-promise").polyfill();Q7:我在函数内用了this.xxx=,为什么抛出Cannot set property ‘xxx’ of undefined;
这又是this的套路了..this是和当前运行的上下文绑定的...
一般你在axios或者其他 promise , 或者setInterval 这些默认都是指向最外层的全局钩子.
简单点说:"最外层的上下文就是 window,vue内则是 Vue 对象而不是实例!";
解决方案:
暂存法: 函数内先缓存 this , let that = this;(let是 es6, es5用 var) 箭头函数: 会强行关联当前运行区域为 this 的上下文;
Q8:我看一些Vue教程有这么些写法,是什么意思@click.prevent,v-demo.a.b;
就拿这两个例子来说吧.
@click.prevent : 事件 修饰符 , 作用就是点击但又阻止默认行为。
v-demo.a.b: 自定义指令 修饰符. 具体看你什么指令了,修饰符的作用大多是给事件增加一些确切的拓展功能,比如阻止事件冒泡,阻止默认行为,访问到原生控件,结合键盘快捷键等等 。
可以自定义修饰符么?也是可以的。可以通过全局 config.keyCodes 对象自定义键值修饰符别名:
Q9:为什么我的引入的小图片渲染出来却是 data:image/png;base64xxxxxxxx
这个是 webpack 里面的对应插件处理的,对于小于多少 K 以下的图片(规定的格式)直接转为 base64格式渲染;
具体配置在webpack.base.conf.js里面的rules里面的 url-loader,这样做的好处:在网速不好的时候先于内容加载和减少http的请求次数来减少网站服务器的负担。
Q10:Component template shold contain exactly one root element.If you are useing v-if on multiple elements , xxxxx
大体就是说,单组件渲染 DOM 区域必须要有一个根元素,不能出现同级元素. 可以用v-if和v-else-if指令来控制其他元素达到并存的状态。
换个直白的解释,就是有一个唯一的父类,包裹者; 比如一个 div(父包含块) 内部多少个同级或者嵌套都行,但是最外层元素不能出现同级元素!
Q11:跨域问题怎么破!比如No ‘Access-Control-Allow-Origin’ header is present on the requested resource.
这种问题老生常谈了,我就不细说了..大体说一下;
1: CORS , 前后端都要对应去配置,IE10 2: nginx 反向代理,一劳永逸
线下开发模式,比如你用了vue-cli, 里面的 webpack 有引入了proxyTable这么个玩意, 也可以做接口反向代理。
// 在 config 目录下的index.jsproxyTable: { "/bp-api": { target: "http://new.d.st.cn", changeOrigin: true, // pathRewrite: { // "^/bp-api": "/" // } }}// target : 就是 api 的代理的实际路径// changeOrigin: 就是是变源,必须是...// pathRewrite : 就是路径重定向,一看就知道当然还有依旧坚挺的jsonp大法!不过局限性比较多,比较适合一些特殊的信息获取!
Q12:我需要遍历的数组值更新了,值也赋值了,为什么视图不更新!
那是因为有局限性啊,官方文档也说的很清楚,只有一些魔改的之后的方法提供跟原生一样的使用姿势(却又可以触发视图更新);
一般我们更常用(除了魔改方法)的手段是使用:this.$set(obj,item,value);
Q13:为什么我的组件间的样式不能继承或者覆写啊!
单组件开发模式下,请确认是否开启了 css模块化功能!也就是scoped(vue-cli 里面配置了,只要加入这个属性就自动启用)。
为什么不能继承或者覆写呢,那时因为每个类或者 id 乃至标签都会给自动在css后面添加hash! 比如:
// 写的时候是这个.trangle{}// 编译过后,加上了 hash.trangle[data-v-1ec35ffc]{}这些都是在 css-loader 里面配置!!!
Q14:路由模式改为history后,除了首次启动首页没报错,刷新访问路由都报错!
必须给对应的服务端配置查询的主页面..也可以认为是主路由入口的引导。
官方文档也有,为毛总有人不喜欢去看文档,总喜欢做伸手党。
Q15:我想拦截页面,或者在页面进来之前做一些事情,可以么?
可以的,各种路由器的钩子!!当然,记忆滚动的位置也可以做到,详情翻翻官网里面的文档。
Q16:TypeError: xxx is not a function
这种问题明显就是写法有问题...能不能动点脑子!!
Q17:Uncaught ReferenceError: xxx is not define
实例内的 data 对应的变量没有声明,你导入模块报这个错误,那绝逼是导出没写好。
Q18:Error in render function:”Type Error: Cannot read property ‘xxx’ of undefined”
这种问题大多都是初始化的姿势不对;比如引入echart这些...仔细去了解下生命周期,再来具体初始化;
vue 组件有时候也会(嵌套组件或者 props传递初始化)..也是基本这个问题
Q19:Unexpected token: operator xxxxx
大佬,这个一看就是语法错误啊,基本都是符号问题, 一般报错会给出哪一行或者哪个组件。
Q20:npm run build之后不能直接访问
大佬!你最起码得在本地搭个服务器才能访问好么!!
Q21:CSSbackground引入图片打包后,访问路径错误
因为打包后图片是在根目录下,你用相对路径肯定报错啊....
你可以魔改 webpack 的配置文件里面的static为./static...但是不建议。你若是把图片什么丢到assets目录下,然后相对路径,打包后是正常的。
Q22:安装模块时命令窗口输出unsupported platform xxx
一般两种情况,node版本不兼容,系统不兼容;
解决方案: 要么不装,要么满足安装要求;
Q23:Unexpected tab charater这些
一般是你用脚手架初始化的时候开了 eslint ;要么遵循规则,要么改变规则;要么直接把 webpack 里面的 eslint 检测给关闭了。
Q24:Failed to mount component: template or render function not defined
组件挂载失败,问题只有这么几个
组件没有正确引入; 挂载点顺序错了了; 自行动手排查
Q25:Unknown custom element: - did you register the component correctly?
组件没有正确引入或者正确使用,依次确认
导入对应的组件在 components 内声明 在 dom 区域声明标签
Q26:axios的 post 请求后台接受不到!
axios默认是 json 格式提交,确认后台是否做了对应的支持;若是只能接受传统的表单序列化,就需要自己写一个转义的方法...当然还有一个更加省事的方案,装一个小模块qs.
npm install qs -S// 然后在对应的地方转就行了..单一请求也行,拦截器也行...我是写在拦截器的.// 具体可以看看我 axios 封装那篇文章//POST传参序列化(添加请求拦截器)Axios.interceptors.request.use( config => { // 在发送请求之前做某件事 if ( config.method === "post" ) { // 序列化 config.data = qs.stringify(config.data); // ***** 这里转义 } // 若是有做鉴权token , 就给头部带上token if (localStorage.token) { config.headers.Authorization = localStorage.token; } return config; }, error => { Message({ // 饿了么的消息弹窗组件,类似toast showClose: true, message: error, type: "error.data.error.message" }); return Promise.reject(error.data.error.message); });Q27:Invalid prop: type check failed for prop “xxx”. Expected Boolean, got String.
这种问题一般就是组件内的 props 类型已经设置了接受的范围类型, 而你传递的值却又不是它需要的类型,写代码严谨些 OK?
Q28:过滤器可以用于DOM区域结合指令么?
// 不行,看下面的错误例子{{item}}// `vue2 `的指令只能用语 mustache`{{}}` , 正确姿势如下:{{ message | capitalize }}Q29:[…Array],…mapState,[SOME_MUTATION] (state) {},increment ({ commit }) {}这种写法是什么鬼!
ES6 (ES2015)的基础去过一遍..上面依次:数组解构,对象解构,对象风格函数,对象解构赋值传递。
Q30:我的 Vue 网站为什么 UC 访问一片空白亦或者flex布局错乱!!
来来来,墙角走起.... UC 号称移动界的 IE 这称号不是白叫的。flexbox 布局错乱,一般是你没有把兼容方案写上..就是带各种前缀,复合属性拆分。
UC访问空白, 有一种情况绝对会造成,那就是 ES6的代码降级不够彻底. 其他情况可能就是路由配置问题(自己去排除),现在的开发都推荐按需引入,靠babel-preset-env 来控制,以达到打包体积减小。
但是这样做的后果,有些内核比较老的...嘿嘿..拜拜。所以最好把代码完全 ES5话!!记住有些特性不能乱使用,没有对应的 polyfill,比如 ES6 的proxy
Q31:this.$set | this.$xxx 这个 $ 是个什么意思?是 jQuery的么,会冲突么?
且看我细细道来.
Vue 的$和 jQuery 的$并没有半毛钱的关系,就跟JavaScript和java一样.Vue 的$是封装了一些 vue 的内建函数,然后导出以$开头...这显然并不是 jQuery的专利;jQuery 的$是选择器!!取得 DOM区域...两者的作用完全不一致!
Q32:Error in event handler for “click”:”xxx”
这个问题大多都是你写的代码有问题.你的事件触发了,但是组件内部缺少对应的实现或者变量,所以抛出事件错误。
解决方案:看着报错慢慢排查
Q33: 组件的通讯有哪几种啊!
基本最常用的是这几种;
父传子: props 子传父: emit兄弟通讯:event bus: 就是找一个中间组件来作为信息传递中介 vuex: 信息树
Q34:vuex的用户信息为什么还要存一遍在浏览器里(sessionStorage or localStorage)
因为 vuex的 store 干不过刷新啊,保存在浏览器的缓存内,若用户刷新的话,值再取一遍呗;
Q35:“有 Vue Vue Router Vuex”或什么”express vue mongodb”的项目学习么
Github 一搜一大堆,提这些问题的人动动脑子!
Q36:线上若是 nginx,如何部署?以及反向代理这些!
1.将node端的服务端口放入服务器的80端口,做反向代理,这里用的是3000端口来做示范
#先定义一个website变量,方便管理以后端口的变更,不会影响到后续的80端口其他的操作upstream website{ server 127.0.0.1:3000;} server { listen 80;#业户逻辑... ...#### location / { proxy_pass http://website; proxy_redirect default ; }#### }Q37:“我会 Vue 我还需要学习 jQuery 或者原生 JS 么”
jQuery还有很多公司在用,源码可以学习的地方很多;原生 js 是根本,不管是哪个前端框架,最终都是 js 实现的;只有基础扎实,才能学的比较深...
框架只是加快开发,提高效率,但不是你在这一行长期立足的根本;
前端的人不仅需要宽度,也要深度...这样才能走的更远....
Q38:npm run dev 报端口错误!Error: listen EADDRINUSE :::8080
自己用 webpack搭脚手架的都不用我说了;Vue-cli 里面的 webpack 配置: config/index.js
dev: { env: require("./dev.env"), port: 8080, // 这里这里,若是这个端口已经给系统的其他程序占用了.改我改我!!!!!! autoOpenBrowser: true, assetsSubDirectory: "static", assetsPublicPath: "/", proxyTable: { "/bp-api": { target: "http://new.d.st.cn", changeOrigin: true, // pathRewrite: { // "^/bp-api": "/" // } }},Q39:什么时候用v-if,什么用 v-show!
我们先来说说两者的核心差异;
v-if : DOM 区域没有生成,没有插入文档..等条件成立的时候才动态插入到页面!有些需要遍历的数组对象或者值,最好用这货控制,等到拿到值才处理遍历,不然一些操作过快的情况会报错,比如数据还没请求到!v-show: DOM 区域在组件渲染的时候同时渲染了,只是单纯用 css 隐藏了,对于下拉菜单,折叠菜单这些数据基本不怎么变动.用这个最合适了..而且可以改善用户体验,因为它不会导致页面的重绘,DOM 操作会!
简言之: DOM结构不怎么变化的用v-show, 数据需要改动很大或者布局改动的用v-if
Q40:是什么,html5的标签么?
你猜对了..html5的标签还真有这么一个,不过 Vue 的 template 有点不一样,不是去给浏览器解析的。
你可以理解为一个临时标签,用来方便你写循环,判断的....
因为最终 template 不会解析到浏览器的页面,他只是在 Vue 解析的过程充当一个包裹层!最终我们看到的是内部处理后的组合的 DOM 结构!
Q41:the “scope” attribute for scoped slots …. replaced by “slot-scope” since 2.5
这个问题只出现老项目升级到 vue2.5 的时候, 提示就是 scope 现在要用 slot-scope 来代替,但是 scope 暂时可以用,以后会移除
Q42:Uncaught ReferenceError : Vue is not defined!
依次排除:
Vue是否正确引入! Vue是否正确实例化! Vue 用的姿势是否正确(比如你直接一个 Vue 的变量!!!刚好又没定义,,具体问题具体分析吧)
Q43:ERROR in static/js/xxxxxxx.js from UglifyJs
我知道其中一种情况会报这种情况,就是你引入的 js,是直接引入压缩版本后的 js(xxx.min.js);然后 webpack 内又启用了 UglifyJs(压缩 JS的), 二重压缩大多都会报错!!
解决方案:引入标准未压缩的 JS
Q44:props不使用:(v-bind)可以传递值么!
可以,只是默认传递的类型会被解析成字符串!若是要传递其他类型,该绑定还是绑定。
Q45:Uncaught TypeError : Cannot set property xxx which has only a getter
这个问题就是你要操作的属性只允许 getter,不允许 setter;
解决方案? 用了别人的东西就要遵循别人的套路来,不然就只能自己动手丰衣足食了!
Q46:单组件中里面的 import xxx from ‘@/components/layout/xxx’中的@是什么鬼!
这是 webpack 方面的知识,看到了也说下吧...webpack可以配置alias(也就是路径别名),玩过 linux 或者 mac 都知道。
依旧如上,会自己搭脚手架的不用我说了...看看 vue-cli 里面的;
文件名: build -> webpack.base.conf.js
resolve: { extensions: [".js", ".vue", ".json"], // 可以导入的时候忽略的拓展名范围 alias: { vue$: "vue/dist/vue.esm.js", "@": resolve("src"), // 这里就是别名了,比如@就代表直接从/src 下开始找起!!! "~": resolve("src/components") }},Q47:SCSS(SASS) 还是 less,stylus 好!
三者都是预处理器;
scss 出现最久,能做的功能比较多,但是若是普通的嵌套写法,继承,mixin 啊,这三个都差不多..会其中一个其他两个的粗浅用法基本也会了,不过写法有些差异:
scss: 写法上是像 css 靠齐sass : 其实也就是 scss , 只是写法不一样...靠的是缩进less : 跟 css 基本靠齐stylus : 一样,靠缩进..跟pug(Jade)一样
使用环境的差异:
scss 可以借助 ruby 或者 node-sass 编译less 可以用 less.js 或者对应的 loader 解析stylus 只能借助 loader 解析,它的出现就是基于 node 的
也有一个后起之秀,主打解耦,插件化的!!! 那就是PostCSS,这个是后处理器! 有兴趣的可以自行去了解,上面的写法都能借助插件实现!
Q48:Failed to compile with x errors : This dependency was not found !
编译错误,对应的依赖没找到! 解决如下:
知道缺少对应的模块,直接装进去,若是一个你已经安装的大模块(比如 axios)里面的子模块(依赖包)出了问题,卸载重装整个大模块.因为你补全不一定有用!
Q49:SyntaxError: Unexpected identifier;
语法错误,看错误信息去找到对应的页面排查!
Q50:为什么我的 npm 或者 yarn 安装依赖会生成 lock文件,有什么用!
lock 文件的作用是统一版本号,这对团队协作有很大的作用;
若是没有 lock 锁定,根据package.json里面的^,~这些.
不同人,不同时间安装出来的版本号不一定一致;
有些包甚至有一些breaking change(破坏性的更新),造成开发很难顺利进行!
Q51: 组件可以缓存么?
可以,用keep-alive;
不过是有代码的..占有内存会多了...所以无脑的缓存所有组件!!!别说性能好了..切换几次,有些硬件 hold不住的,浏览器直接崩溃或者卡死..
所以keep-alive一般缓存都是一些列表页,不会有太多的操作,更多的只是结果集的更换..给路由的组件meta增加一个标志位,结合v-if就可以按需加上缓存了!
Q52:package.json里面的dependencies 和devDependencies的差异!
其实不严格的话,没有特别的差异;若是严格,遵循官方的理解;
dependencies : 存放线上或者业务能访问的核心代码模块,比如 vue,vue-router;
devDependencies: 处于开发模式下所依赖的开发模块,也许只是用来解析代码,转义代码,但是不产生额外的代码到生产环境, 比如什么babel-core这些 如何把包安装到对应的依赖下呢?
npm install --save xxxx // dependenciesnpm install --save-dev xxxx // devDependencies//也能用简易的写法(i:install,-S:save,-D:save-dev)npm i -S xxxx // npm install --save xxxxnpm i -D xxxx // npm install --save-dev xxxxQ53:安装chromedriver报错!!姿势没错啊npm i -D chromedriver
恩,伟大的 GFW.....解决方案:指定国内的源安装就可以了
npm install --save-dev chromedriver --chromedriver_cdnurl=http://cdn.npm.taobao.org/dist/chromedriverQ54:Vue ,react, Angular学习哪个好?哪个工作比较好找!
Vue属于渐进式开发,传统开发过渡 MVVM 模式的小伙伴,Vue 比较好上手,学习成本比较低。
基础比较好的,有折腾精神的,可以选择NG5或者react 16;
NG5需要学习typescript和rxjs,还用到比较多的新东西,比如装饰器,后端的注入概念.ng有自己的一整套 MVVM 流程;
而Vue和React核心只是view,可以搭配自己喜欢的
React的写法偏向函数式写法,还有 jsx,官方自己有 flow,当然也能搭配ts,我也没怎么接触..所以也有一定的学习成本;
至于哪个比较好找工作!!!告诉你..若是只会一个框架,那不是一个合格的前端;
人家要的是动手能力,解决能力!!!!技术和待遇是成正比的!!
颜值和背景,学历,口才可以加分..但是这些条件你必须要有的基础下才能考虑这些!!!
Q55:我有个复杂组件需要有新增和编辑的功能同时存在,但是字段要保持不变性怎么破
字段保持不变性怎么理解呢? 就是说比如新增和编辑同时共享一份 data;
有一种就是路由变了,组件渲染同一个(不引起组件的重新渲染和销毁!),但是功能却不同(新增和编译)..
比如从编辑切到新增,data必须为空白没有赋值的,等待我们去赋值;
这时候有个东西就特别适合了,那就是immutable-js;
这个东西可以模拟数据的唯一性!或者叫做不变性!
Q56:“首屏加载比较慢!!怎么破!打包文件文件比较大”
依次排除和确认:
减少第三方库的使用,比如jquey这些都可以不要了,很少操作 dom,而且原生基本满足开发。
若是引入moment这些,webpack 排除国际化语言包。
webpack 常规压缩js,css, 愿意折腾的还可以引入 dll 这些。
路由组件采用懒加载。
加入路由过渡和加载等待效果,虽然不能解决根本,但起码让人等的舒心一点不是么!!!
整体下来,打包之后一般不会太大;
但是倘若想要更快?那就只能采用服务端渲染(SSR)了,可以避免浏览器去解析模板和指令这些;
直接返回一个 html ....还能 seo...
Q57: Vue SPA 没法做优化(SEO)!有解决方案么
可以的,SSR(服务端渲染就能满足你的需求),因为请求回来就是一个处理完毕的 html,现在 vue 的服务端开发框架有这么个比较流行,如下Nuxt.js。
Q58:Vue可以写 hybird App 么!
当然可以,两个方向.
codorva nativescriptWeexQ59: Vue可以写桌面端么?
当然可以,有electron和node-webkit(nw); 我只了解过electron;
electronelectron-vue: Vue-cli 针对 electron 的脚手架模板Q60:Vue开发,项目中还需要 jQuery么
分情况探讨:
若是老项目,只是单纯引入 Vue 简化开发的,依旧用吧…重构项目?或者发起新项目的,真心没必要了.开发思路不一样,很多以前用 DOM 操作的现在基本可以数据驱动实现,而少量迫不得已的DOM 操作原生就能搞定…而且能小打包体积,速度又快,何乐而不为!
Vue 面试中常问知识点整理
生命周期:Vue实例从开始创建、初始化数据、编译模板、挂载Dom→渲染、更新→渲染、卸载等一系列过程,我们称这是Vue的生命周期,各个阶段有相对应的事件钩子。beforeCreate(创建前),在数据观测和初始化事件还未开始
created(创建后),完成数据观测,属性和方法的运算,初始化事件, $el 属性还没有显示出来
beforeMount(载入前),在挂载开始之前被调用,相关的render函数首次被调用。实例已完成以下的配置:编译模板,把data里面的数据和模板生成html。注意此时还没有挂载html到页面上。
mounted(载入后),在 el 被新创建的 vm.$el 替换,并挂载到实例上去之后调用。实例已完成以下的配置:用上面编译好的html内容替换 el 属性指向的DOM对象。完成模板中的html渲染到html页面中。此过程中进行ajax交互。
beforeUpdate(更新前),在数据更新之前调用,发生在虚拟DOM重新渲染和打补丁之前。可以在该钩子中进一步地更改状态,不会触发附加的重渲染过程。
updated(更新后),在由于数据更改导致的虚拟DOM重新渲染和打补丁之后调用。调用时,组件DOM已经更新,所以可以执行依赖于DOM的操作。然而在大多数情况下,应该避免在此期间更改状态,因为这可能会导致更新无限循环。该钩子在服务器端渲染期间不被调用。
beforeDestroy(销毁前),在实例销毁之前调用。实例仍然完全可用。
destroyed(销毁后),在实例销毁之后调用。调用后,所有的事件监听器会被移除,所有的子实例也会被销毁。该钩子在服务器端渲染期间不被调用。
注意:
created 阶段的ajax请求与 mounted 请求的区别:前者页面视图未出现,如果请求信息过多,页面会长时间处于白屏状态。
mounted 不会承诺所有的子组件也都一起被挂载。如果你希望等到整个视图都渲染完毕,可以用 vm.$nextTick 。
初始化组件时,仅执行了 beforeCreate/Created/beforeMount/mounted 四个钩子函数
当改变 data 中定义的变量(响应式变量)时,会执行 beforeUpdate/updated 钩子函数
当切换组件(当前组件未缓存)时,会执行 beforeDestory/destroyed 钩子函数
初始化和销毁时的生命钩子函数均只会执行一次, beforeUpdate/updated 可多次执行
仅当子组件完成挂载后,父组件才会挂载
当子组件完成挂载后,父组件会主动执行一次beforeUpdate/updated钩子函数(仅首次)
父子组件在data变化中是分别监控的,但是在更新props中的数据是关联的
销毁父组件时,先将子组件销毁后才会销毁父组件
组件的初始化(mounted之前)分开进行,挂载是从上到下依次进行
当没有数据关联时,兄弟组件之间的更新和销毁是互不关联的
mixin中的生命周期与引入该组件的生命周期是仅仅关联的,且mixin的生命周期优先执行
1、什么是vue生命周期?
答: Vue 实例从创建到销毁的过程,就是生命周期。从开始创建、初始化数据、编译模板、挂载Dom→渲染、更新→渲染、销毁等一系列过程,称之为 Vue 的生命周期。
2、vue生命周期的作用是什么?
答:它的生命周期中有多个事件钩子,让我们在控制整个Vue实例的过程时更容易形成好的逻辑。
3、vue生命周期总共有几个阶段?
答:它可以总共分为8个阶段:创建前/后、载入前/后、更新前/后、销毁前/销毁后。
5、DOM 渲染在哪个周期中就已经完成?
答:DOM 渲染在 mounted 中就已经完成了。
vue实现数据双向绑定主要是:采用数据劫持结合发布者-订阅者模式的方式,通过 Object.defineProperty() 来劫持各个属性的 setter,getter ,在数据变动时发布消息给订阅者,触发相应监听回调。当把一个普通 Javascript 对象传给 Vue 实例来作为它的 data 选项时,Vue 将遍历它的属性,用 Object.defineProperty() 将它们转为 getter/setter 。用户看不到 getter/setter,但是在内部它们让 Vue 追踪依赖,在属性被访问和修改时通知变化。
vue的数据双向绑定 将MVVM作为数据绑定的入口,整合 Observer ,Compile和 Watcher 三者,通过 Observer 来监听自己的 model 的数据变化,通过 Compile 来解析编译模板指令(vue中是用来解析 {{}}),最终利用 watcher 搭起observer和 Compile 之间的通信桥梁,达到数据变化 —>视图更新;视图交互变化( input )—>数据 model 变更双向绑定效果。
js实现简单的双向绑定:
1、父组件与子组件传值
父组件传给子组件:子组件通过 props 方法接受数据;
子组件传给父组件: $emit 方法传递参数
2、非父子组件间的数据传递,兄弟组件传值 eventBus ,就是创建一个事件中心,相当于中转站,可以用它来传递事件和接收事件。项目比较小时,用这个比较合适。
hash模式:在浏览器中符号“#”,#以及#后面的字符称之为hash,用 window.location.hash 读取。特点:hash虽然在URL中,但不被包括在HTTP请求中;用来指导浏览器动作,对服务端安全无用,hash不会重加载页面。
history模式:history采用HTML5的新特性;且提供了两个新方法: pushState(), replaceState()可以对浏览器历史记录栈进行修改,以及popState事件的监听到状态变更。
需求一:
在一个列表页中,第一次进入的时候,请求获取数据。
点击某个列表项,跳到详情页,再从详情页后退回到列表页时,不刷新。
也就是说从其他页面进到列表页,需要刷新获取数据,从详情页返回到列表页时不要刷新。
解决方案
在 App.vue设置:
假设列表页为 list.vue ,详情页为 detail.vue ,这两个都是子组件。
我们在 keep-alive 添加列表页的名字,缓存列表页。
然后在列表页的 created 函数里添加ajax请求,这样只有第一次进入到列表页的时候才会请求数据,当从列表页跳到详情页,再从详情页回来的时候,列表页就不会刷新。
这样就可以解决问题了。
需求二:
在需求一的基础上,再加一个要求:可以在详情页中删除对应的列表项,这时返回到列表页时需要刷新重新获取数据。
我们可以在路由配置文件上对 detail.vue 增加一个 meta 属性。
这个 meta 属性,可以在详情页中通过 this.$route.meta.isRefresh 来读取和设置。
设置完这个属性,还要在 App.vue 文件里设置 watch 一下 $route 属性。
这样就不需要在列表页的 created 函数里用 ajax 来请求数据了,统一放在 App.vue 里来处理。
触发请求数据有两个条件:
从其他页面(除了详情页)进来列表时,需要请求数据。
从详情页返回到列表页时,如果详情页 meta 属性中的 isRefresh 为 true ,也需求重新请求数据。
当我们在详情页中删除了对应的列表项时,就可以将详情页 meta 属性中的 isRefresh 设为 true 。这时再返回到列表页,页面会重新刷新。
1、css只在当前组件起作用
答:在 style 标签中写入 scoped 即可 例如: <style scoped></style>
2、v-if 和 v-show 区别
答: v-if 按照条件是否渲染, v-show 是 display 的 block 或 none ;
3、 $route 和 $router 的区别
答: $route 是“路由信息对象”,包括 path,params,hash,query,fullPath,matched,name 等路由信息参数。而 $router 是“路由实例”对象包括了路由的跳转方法,钩子函数等。
vue常见面试题
1.什么是vue的生命周期?Vue实例从创建到销毁的过程,就是生命周期。也就是从开始创建、初始化数据、编译模板、挂载DOM->渲染、更新->渲染、卸载等一系列过程,我们称这是Vue的生命周期。
2.vue生命周期的作用是什么?
它的生命周期中有多个事件钩子,让我们在控制整个vue实例的过程时更容易形成好的逻辑。
3.Vue生命周期总共有几个阶段?
它可以总共分为8个阶段:创建前/后,载入前/后,更新前/后,销毁前/销毁后
4.第一次页面加载会触发那几个钩子?
第一次页面加载时会触发beforeCreate,created,beforeMount,mounted
5.DOM渲染在哪个周期中就已经完成?
DOM渲染在mounted中就已经完成了
6.生命周期钩子的一些使用方法:
1.beforecreate:可以在加个loading事件,在加载实例是触发
2.created:初始化完成时的事件写在这里,如在这结束loading事件,异步请求也适宜在这里调用
3.mounted:挂载元素,获取到dom节点
4.updated:如果对数据统一处理,在这里写上相应函数
5.beforeDestroy:可以一个确认停止事件的确认框
6.nextTick:更新数据后立即操作dom
7.v-show与v-if的区别
v-show是css切换,v-if是完整的销毁和重新创建
使用频繁切换时用v-show,运行时较少改变时用v-if
V-if=’false’v-if是条件渲染,当false的时候不会渲染
使用v-if的时候,如果值为false,那么页面将不会有这个html标签生成
v-show则是不管值是为true还是false,html元素都会存在,只是css中的display显示或隐藏
v-show 仅仅控制元素的显示方式,将 display 属性在 block 和 none 来回切换;而v-if会控制这个 DOM 节点的存在与否。当我们需要经常切换某个元素的显示/隐藏时,使用v-show会更加节省性能上的开销;当只需要一次显示或隐藏时,使用v-if更加合理。
8.开发中常用的指令有哪些?
v-model:一般用在表达输入,很轻松的实现表单控件和数据的双向绑定
v-html:更新元素的innerHTML
v-show与v-if:条件渲染,注意二者区别
v-on:click:可以简写为@click,@绑定一个事件。如果事件触发了,就可以指定事件的处理函数
v-for:基于源数据多次渲染元素或模板
v-bind:当表达式的值改变时,将其产生的连带影响,响应式地作用于DOM语法
v-bind:title=”msg”简写:title="msg"
9.绑定class的数组用法
1.对象方法v-bind:class="{orange:isRipe, green:isNotRipe}”
2.数组方法v-bind:class="[class1,class2]"
3.行内v-bind:style="{color:color,fontSize:fontSize+px}”
10.路由跳转方式
1.router-link标签会渲染为标签,咋填template中的跳转都是这种;
2.另一种是编辑是导航,也就是通过js跳转比如router.push(/home)
12. computed和watch有什么区别
computed
computed是计算属性,也就是计算值,它更多用于计算值的场景
computed具有缓存性,computed的值在getter执行后是会缓存的,只有在它依赖的属性值改变之后,下一次获取computed的值时重新调用对应的getter来计算
computed适用于计算比较消耗性能的计算场景
watch
watch更多的是[观察]的作用,类似于某些数据的监听回调,用于观察props $emit或者本组件的值,当数据变化时来执行回调进行后续操作
无缓存性,页面重新渲染时值不变化也会执行
小结
当我们要进行数值计算,而且依赖于其他数据,那么把这个数据设计为computed
如果你需要在某个数据变化时做一些事情,使用watch来观察这个数据变化。
13. vue组件的scoped属性的作用
在style标签上添加scoped属性,以表示它的样式作用于当下的模块,很好的实现了样式私有化的目的;
但是也得慎用:样式不易(可)修改,而很多时候,我们是需要对公共组件的样式做微调的;
解决办法:
①:使用混合型的css样式:(混合使用全局跟本地的样式) /* 全局样式 */ /* 本地样式 */
②:深度作用选择器(>>>)如果你希望 scoped 样式中的一个选择器能够作用得“更深”,例如影响子组件,你可以使用 >>> 操作符: .a >>> .b { /* ... */ }
14. vue是渐进式的框架的理解:( 主张最少,没有多做职责之外的事 )
Vue的核心的功能,是一个视图模板引擎,但这不是说Vue就不能成为一个框架。如下图所示,这里包含了Vue的所有部件,在声明式渲染(视图模板引擎)的基础上,我们可以通过添加组件系统、客户端路由、大规模状态管理来构建一个完整的框架。更重要的是,这些功能相互独立,你可以在核心功能的基础上任意选用其他的部件,不一定要全部整合在一起。可以看到,所说的“渐进式”,其实就是Vue的使用方式,同时也体现了Vue的设计的理念
在我看来,渐进式代表的含义是:主张最少。视图模板引擎
每个框架都不可避免会有自己的一些特点,从而会对使用者有一定的要求,这些要求就是主张,主张有强有弱,它的强势程度会影响在业务开发中的使用方式。
比如说,Angular,它两个版本都是强主张的,如果你用它,必须接受以下东西:
必须使用它的模块机制- 必须使用它的依赖注入- 必须使用它的特殊形式定义组件(这一点每个视图框架都有,难以避免)
所以Angular是带有比较强的排它性的,如果你的应用不是从头开始,而是要不断考虑是否跟其他东西集成,这些主张会带来一些困扰。
Vue可能有些方面是不如React,不如Angular,但它是渐进的,没有强主张,你可以在原有大系统的上面,把一两个组件改用它实现,当jQuery用;也可以整个用它全家桶开发,当Angular用;还可以用它的视图,搭配你自己设计的整个下层用。也可以函数式,都可以,它只是个轻量视图而已,只做了自己该做的事,没有做不该做的事,仅此而已。
渐进式的含义,我的理解是:没有多做职责之外的事。
15.vue.js的两个核心是什么(数据驱动、组件系统。)
数据驱动:Object.defineProperty和存储器属性: getter和setter(所以只兼容IE9及以上版本),可称为基于依赖收集的观测机制,核心是VM,即ViewModel,保证数据和视图的一致性。
16.vue常用修饰符
修饰符分为:一般修饰符,事件修身符,按键、系统
①一般修饰符:
.lazy:v-model 在每次 input 事件触发后将输入框的值与数据进行同步 。你可以添加 lazy 修饰符,从而转变为使用 change 事件进行同步
<inputv-model.lazy="msg">
.number
<inputv-model.number="age"type="number">
.trim
1.如果要自动过滤用户输入的首尾空白字符<inputv-model.trim=trim>
② 事件修饰符
<av-on:click.stop="doThis"></a><!-- 阻止单击事件继续传播 -->
<formv-on:submit.prevent="onSubmit"></form> <!-- 提交事件不再重载页面 -->
<av-on:click.stop.prevent="doThat"></a> <!-- 修饰符可以串联 -->
<formv-on:submit.prevent></form> <!-- 只有修饰符 -->
<divv-on:click.capture="doThis">...</div> <!-- 添加事件监听器时使用事件捕获模式 --> <!-- 即元素自身触发的事件先在此处处理,然后才交由内部元素进行处理 -->
<divv-on:click.self="doThat">...</div> <!-- 只当在 event.target 是当前元素自身时触发处理函数 --> <!-- 即事件不是从内部元素触发的 -->
<av-on:click.once="doThis"></a> <!-- 点击事件将只会触发一次 -->
③按键修饰符
全部的按键别名:
.enter.tab.delete(捕获“删除”和“退格”键).esc.space.up.down.left.right.ctrl.alt.shift.meta
<inputv-on:keyup.enter="submit">或者<input@keyup.enter="submit">
④系统修饰键 (可以用如下修饰符来实现仅在按下相应按键时才触发鼠标或键盘事件的监听器。)
.ctrl.alt.shift.meta
<input@keyup.alt.67="clear">或者<div@click.ctrl="doSomething">Dosomething</div><!-- Ctrl + Click -->
20.Vue 组件中 data 为什么必须是函数
vue组件中data值不能为对象,因为对象是引用类型,组件可能会被多个实例同时引用。
如果data值为对象,将导致多个实例共享一个对象,其中一个组件改变data属性值,其它实例也会受到影响。
文章评论