Vuejs的一些总结

来源:互联网 发布:全景补地软件 编辑:程序博客网 时间:2024/06/07 16:08

本文基于vue1.x

基于vue2.x&webpack2.x请移步至

Vue2.x踩坑与总结
Webpack2.x踩坑与总结

最近一段时间忙着工作的一些事情,同时自己也在试着把项目中的一些移动端页面试着用vuejs重写,所以没时间写文章,今天终于有空可以写一下,由于页面并没有写完,所以就将自己这几天做页面的时候遇到的一些问题总结了一下。其实很多在官网都能找到,但是我们只看官网教程不去写,很难理解到底是什么意思,这里我把我用到的列出来。

文章中提到的很多东西都在我的demo中用到,我的demo地址:https://github.com/MrZhang123...

1.Vuejs组件

vuejs构建组件使用

Vue.component('componentName',{ /*component*/ });

这里注意一点,组件要先注册再使用,也就是说:

Vue.component('mine',{           template:'#mineTpl',           props:['name','title','city','content']        }); var v=new Vue({      el:'#vueInstance',      data:{          name:'zhang',          title:'this is title',         city:'Beijing',         content:'these are some desc about Blog'     }});

如果反过来会报错,因为反过来代表先使用了组件的,但是组件却没注册。

webpack报错后,使用webpack --display-error-details可以排错

2.指令keep-alive

在看demo的时候看到在vue-router写着keep-alivekeep-alive的含义:
如果把切换出去的组件保留在内存中,可以保留它的状态或避免重新渲染。为此可以添加一个keep-alive指令

<component :is='curremtView' keep-alive></component>

3.如何让css只在当前组件中起作用

在每一个vue组件中都可以定义各自的css,js,如果希望组件内写的css只对当前组件起作用,只需要在style中写入scoped,即:

<style scoped></style>

4.vuejs循环插入图片

在写循环的时候,写入如下代码:

<div class="bio-slide" v-for="item in items">       <img src="{{item.image}}"></div>

此时在控制台会出现警告
[Vue Warn]: src="{{item.image}}": interpolation in "src" attribute will cause a 404 request. Use v-bind:src instead.这里意思是在“src”属性插值将导致404请求。使用v-bind:src代替。
所以替换成如下:

<div class="bio-slide" v-for="item in items">       <img v-bind:src="item.image"></div>

这里需要主要,v-bind在写的时候不能再用{{}},根据官方的说法:

<a v-bind:href="url"></a>

这里 href 是参数,它告诉 v-bind 指令将元素的 href 特性跟表达式 url 的值绑定。可能你已注意到可以用特性插值href="{{url}}" 获得同样的结果:这样没错,并且实际上在内部特性插值会转为 v-bind 绑定。

5.绑定value到Vue实例的一个动态属性上

对于单选按钮,勾选框及选择框选项,v-model绑定的value通常是静态字符串(对于勾选框是逻辑值):

<!-- `toggle` 为 true 或 false --><input type="checkbox" v-model="toggle">

但是有时候想绑定value到vue实例的一个动态属性上,这时可以用v-bind实现,并且这个属性的值可以不是字符串。例如绑定Checkbox的value到vue实例的一个动态属性:

<input     type="checkbox"    v-model="toggle"    v-bind:true-value="a"    v-bind:false-value="b"><p>{{toggle}}</p>

这里绑定后,并不是说就可以点击后由true,false的切换变为a,b的切换,因为这里定义的动态a,b是scope上的a,b,并不能直接显示出来,此时

//当选中时vm.toggle === vm.a//当没选中时vm.toggle === vm.b

所以此时需要在data中定义a,b,即:

new Vue({    el:'...',    data:{        a:'a',        b:'b'        }});

6.片段实例

下面几种情况会让实例变成一个片断实例:

  1. 模板包含多个顶级元素。

  2. 模板只包含普通文本。

  3. 模板只包含其它组件(其它组件可能是一个片段实例)。

  4. 模板只包含一个元素指令,如<partial> 或 vue-router 的 <router-view>

  5. 模板根节点有一个流程控制指令,如 v-if  v-for

这些情况让实例有未知数量的顶级元素,它将把它的 DOM 内容当作片断。片断实例仍然会正确地渲染内容。不过,它没有一个根节点,它的$el 指向一个锚节点,即一个空的文本节点(在开发模式下是一个注释节点)。
但是更重要的是,组件元素上的非流程控制指令,非 prop 特性和过渡将被忽略,因为没有根元素供绑定:

<!-- 不可以,因为没有根元素 --><example v-show="ok" transition="fade"></example> <!-- props 可以 --><example :prop="someData"></example> <!-- 流程控制可以,但是不能有过渡 --><example v-if="ok"></example>

片段实例也有用处,但是通常情况下组件有一个根节点比较好,它会保证组件元素上的指令和特性能正确的转换,同时性能也稍微好些。

7.路由嵌套

路由嵌套会将其他组件渲染到该组件内,而不是进行整个页面跳转router-view本身就是将组件渲染到该位置,想要进行页面跳转,就要将页面渲染到根组件,在起始配置路由时候写到:

var App = Vue.extend({ root });router.start(App,'#app');

这里首先将根组件注册进来,用于将路由中配置好的各个页面渲染出来,然后将根组件挂载到与#app匹配的元素上。

8.实现多个根据不同条件显示不同文字的方法

v-if,v-else可以实现条件选择,但是如果是多个连续的条件选择,则需要用到计算属性computed。例如实现当输入框中什么都没写的时候显示字符串‘empty’,否则显示输入框中的内容,代码如下:

<div id="test">  <input type="text" v-model="inputValue">  <h1>{{changeVaule}}</h1></div>
new Vue({  el:'#test',  data:{    changeVaule:'123'  },  computed :{    changeVaule:function(){      if(this.inputValue!==''){        return this.inputValue;      }else{        return 'empty';      }    }  }});

9.Vuejs在变化检测问题

1.检测数组

由于javascript的限制,vuejs不能检测到下面数组的变化:

  1. 直接索引设置元素,如vm.item[0]={};

  2. 修改数据的长度,如vm.item.length

为了解决问题1,Vuejs扩展了观察数组,为它添加一个$set()方法:

// 与 `example1.items[0] = ...` 相同,但是能触发视图更新example1.items.$set(0, { childMsg: 'Changed!'})

问题2,需要一个空数组替换items

除了$set(),vuejs也为观察数组添加了$remove()方法,用于从目标数组中查找并删除元素,在内部调用了splice()。因此,不必:

var index = this.items.indexOf(item)if (index !== -1) {  this.items.splice(index, 1)}

只需:

this.items.$remove(item);

2.检测对象

受ES5的显示,Vuejs不能检测到对象属性的添加或删除。因为Vuejs在初始化时候将属性转化为getter/setter,所以属性必须在data对象才能让Vuejs转换它,才能让它是响应的,例如:

var data = { a: 1 }var vm = new Vue({  data: data})// `vm.a` 和 `data.a` 现在是响应的 vm.b = 2// `vm.b` 不是响应的 data.b = 2// `data.b` 不是响应的

不过,有办法在实例创建之后添加属性并且让它是响应的。对于Vue实例,可以使用$set(key,value)实例方法:

vm.$set('b', 2)// `vm.b` 和 `data.b` 现在是响应的

对于普通数据对象,可以使用全局方法Vue.set(object, key, value):

Vue.set(data, 'c', 3)// `vm.c` 和 `data.c` 现在是响应的

有时你想向已有对象上添加一些属性,例如使用 Object.assign() 或 _.extend() 添加属性。但是,添加到对象上的新属性不会触发更新。这时可以创建一个新的对象,包含原对象的属性和新的属性:

// 不使用 `Object.assign(this.someObject, { a: 1, b: 2 })`this.someObject = Object.assign({}, this.someObject, { a: 1, b: 2 })

10.关于vuejs页面闪烁{{message}}

在vuejs指令中有v-cloak,这个指令保持在元素上直到关联实例结束编译。和CSS规则如[v-cloak]{display:none}一起用时,这个指令可以隐藏未编译的Mustache标签直到实例准备完毕。用法如下:

[v-cloak]{    display:none;}
<div v-cloak>{{message}}</div>

这样<div>不会显示,直到编译结束

11.关于在v-for循环时候v-model的使用

有时候需要循环生成input,用v-model绑定后,利用vuejs操作它,此时我们可以在v-model中写一个数组selected[$index],这样就可以给不同的input绑定不同的v-model,从而分别操作他们。这个我在demo中的dataBind.vue中用到。

12.vuejs中过渡动画

在vuejs中,css定义动画:

  .zoom-transition{        width:60%;        height:auto;        position: absolute;        left:50%;        top:50%;        transform: translate(-50%,-50%);        -webkit-transition: all .3s ease;        transition: all .3s ease;    }    .zoom-enter, .zoom-leave{        width:150px;        height:auto;        position: absolute;        left:20px;        top:20px;        transform: translate(0,0);    }

其中动画在定的时候要注意上下对应,上面有什么,下面有什么,都要变化的,如果有不变化的,应该抽离出去,作为公共css样式,在上面的css中,如果我只写 transform: translate(-50%,-50%);而不写下面的transform: translate(0,0);则会导致上面的transform: translate(-50%,-50%);被添加到下面,认为这个是不变的。

13.指令v-el的使用

有时候我们想就像使用jquery那样去访问一个元素,此时就可以使用v-el指令,去给这个元素注册一个索引,方便通过所属实例的$el访问这个元素。
注意
HTML不区分大小写,所以v-el:someEl将转换为全小写。可以用v-el:some-el然后设置this.$el.someEl
示例

<span v-el:msg>hello</span><span v-el:other-msg>world</span>
this.$els.msg.textContent // -> "hello"this.$els.otherMsg.textContent // -> "world"this.$els.msg //-><span>hello</span>

14.关于vuejs中使用事件名

在vuejs中,我们经常要绑定一些事件,有时候给DOM元素绑定,有时候给组件绑定。绑定事件在HTML中用v-on:click-"event",这时evet的名字不要出现大写,因为在1.x中不区分大小写,所以如果我们在HTML写v-on:click="myEvent"而在js中写myEvent就出错误,所以在vuejs的1.x绑定事件时候,要尽量避免使用大写字母。在2.0中没有该限制!

15.v-if与v-show的区别

v-if直接不渲染这个DOM元素,而v-show是会渲染DOM元素,只是使用display:none隐藏,打开开发者工具可以看到该DOM

16.关于transition全局钩子如何在组件中使用

Vue.transition是定义一个全局transition钩子的,如果想针对组件定义,则需要如下写法:

export default{    transition:{        'fade':{            enter() {},            leave() {}       }   }}

这样fade这个过度钩子只会作用于组件内,如果同时有同名的全局钩子,则会优先使用组建定义的

17.利用vue-router如何实现组件在渲染出来前执行某个事件

export default{    data(){        return{            selected:0,            currentView:'view_0'        }    },    methods:{        choose(index) {            this.selected=index;            this.currentView='view_'+index;        }    },    route:{        data() {            /*每次切换路由,在渲染出页面前都会执行*/        }    }

}

vue2.x

1.独立构建vs运行时构建

在按照vue1.0的配置配置好webpack后,会出现Failed to mount component: template or render function not defined. (found in root instance) 的错误,这里涉及到vue2.0与vue1.0的第一个不同的地方。具体区别独立构建 vs 运行时构建。解决方法为在webpack配置文件中添加如下配置项:

resolve: {  alias: {    'vue$': 'vue/dist/vue.common.js'  }}

2.挂载点的选择

在原来的vue1.0的项目中我使用body元素作为挂载点,但是在vue2.0中,如果使用body或者html作为挂载点,则会报以下警告:Do not mount Vue to <html> or <body> - mount to normal elements instead.

<font color='red'>在vue1.0中允许开发者以body或者html作为根实体的挂载点,但是到了2.0后,只能通过独立的节点挂载,例如:div等,否则报警告</font>

3.动态组件渲染(跟1.x类似)

多个组件可以使用同一个挂载点,然后动态地在它们之间切换。使用保留的 <component> 元素,动态地绑定到它的 is 特性:

<!-- 动态组件由 vm 实例的属性值 `componentId` 控制 --><component :is="componentId"></component><!-- 也能够渲染注册过的组件或 prop 传入的组件 --><component :is="$options.components.child"></component>

keep-alive

如果把切换出去的组件保留在内存中,可以保留它的状态或避免重新渲染。为此可以添加一个 keep-alive 指令参数:

<keep-alive>  <component :is="currentView">    <!-- 非活动组件将被缓存! -->  </component></keep-alive>

4.ref

有时候需要直接在父组件中访问子组件实例,或者直接操作DOM元素,此时需要使用ref

ref被用来给元素或子元素注册引用信息。引用信息会根据父组件的$refs对象进行注册。如果在普通的DOM元素上使用,引用信息就是元素,如果用在子组件上,引用信息就是组件实例。

<!-- vm.$refs.p will be the DOM node --><p ref="p">hello</p><!-- vm.$refs.child will be the child comp instance --><child-comp ref="child"></child-comp>

当 v-for 用于元素或组件的时候,引用信息将是包含DOM节点或组件实例数组。

关于ref注册时间的重要说明: 因为ref本身是作为渲染结果被创建的,在初始渲染的时候你不能访问它们 - 它们还不存在!$refs 也不是响应式的,因此你不应该试图用它在模版中做数据绑定。

5.自定义事件

在vue自定义事件使用$on$emit,前者用于触发监听,后者用于触发,监听可以有两种方式

<!--使用v-on在html中监听--><my-component v-on:test="callbackFun"></my-component><script>  //直接用$on监听  vm.$on('text',function(){})</script>

6.组件命名的约定

当注册组件(或者 props)时,可以使用 kebab-case ,camelCase ,或 TitleCase

// 在组件定义中components: {  // 使用 kebab-case 形式注册  'kebab-cased-component': { /* ... */ },  // register using camelCase  'camelCasedComponent': { /* ... */ },  // register using TitleCase  'TitleCasedComponent': { /* ... */ }}

在 HTML 模版中,只能使用 kebab-case 形式:

<!-- 在HTML模版中始终使用 kebab-case --><kebab-cased-component></kebab-cased-component><camel-cased-component></camel-cased-component><title-cased-component></title-cased-component>

当使用字符串模式时可以使用 camelCase 、 TitleCase 或者 kebab-case 来引用:

<!-- 在字符串模版中可以用任何你喜欢的方式! --><my-component></my-component><myComponent></myComponent><MyComponent></MyComponent>

7.子组件中使用this

有时候子组件简单,可以在父组件中直接注册,此时在子组件内使用this就是子组件实例并不是父组件,例如:

export default{    data(){        return{            parentMsg:'hello!'        }    },    components:{        child:{            props:['inputMessage'],            template:'<span>{{inputMessage}}</span>'        },        'child-secound':{            props:['inputMessage'],            template:'<span>{{upperCase}}</span>',            computed:{                upperCase(){                    return this.inputMessage.toUpperCase();                }            }            }    }}

8.key的使用

一般情况下,vue在渲染完成后,如果数据发生变化,只会重新渲染数据,不会重新渲染整个元素,但是有时候我们需要元素被重新渲染,此时就需要使用key关键字,使用v-bind绑定key关键字,可以实现在数据发生变化时候重新渲染整个元素。注:同一父级元素下所有子元素如果都要在数据变化后重新渲染元素,则需要被绑定的key

9.v-move的使用

在使用<transition-group>时候,不仅可以定义进入离开动画,还可以使用新增的v-move特性,与过渡一样,默认为v-move,可以用name进行自定义前缀,也可以用move-class属性手动设定。用了这个之后就可以实现移动过程中的动画。

10.跳过css检测

对于只使用js过度的元素使用v-bind:css="false"跳过vue对css的检测。

render函数的使用

createElement接受三个参数:

  • {String | Object | Function}即一个HTML标签 | 组件选项 | 一个函数,必须返回上述其中一个

  • {Object}一个对应HTML标签属性的数据对象(可选)

  • {String | Array}子节点(VNode)(可选)

⚠️ 关于第三个参数的说明

  • createElement第三个参数,如果是String,则类似于innerHTML,如果是Array,则可以写入一个执行函数,这个函数用于创建另一个DOM结构(而且这里如果想写入一个执行函数,必须是数组!!)

  • 每个createElement只能创建一个元素,所以如果是创建多个元素相互嵌套,需要多个createElement函数相互嵌套,最后再render,这个跟原生js创建DOM元素类似

  • 如果需要同时渲染多个元素,则需要在第三个参数的数组中,分别写入需要渲染的元素,此时Vue会按照数组中顺序进行渲染

完整数据对象:

{  // 和`v-bind:class`一样的 API  'class': {    foo: true,    bar: false  },  // 和`v-bind:style`一样的 API  style: {    color: 'red',    fontSize: '14px'  },  // 正常的 HTML 特性  attrs: {    id: 'foo'  },  // 组件 props  props: {    myProp: 'bar'  },  // DOM 属性  domProps: {    innerHTML: 'baz'  },  // 事件监听器基于 "on"  // 所以不再支持如 v-on:keyup.enter 修饰器  // 需要手动匹配 keyCode。  on: {    click: this.clickHandler  },  // 仅对于组件,用于监听原生事件,而不是组件使用 vm.$emit 触发的事件。  nativeOn: {    click: this.nativeClickHandler  },  // 自定义指令. 注意事项:不能对绑定的旧值设值  // Vue 会为您持续追踨  directives: [    {      name: 'my-custom-directive',      value: '2'      expression: '1 + 1',      arg: 'foo',      modifiers: {        bar: true      }    }  ],  // Scoped slots in the form of  // { name: props => VNode | Array<VNode> }  scopedSlots: {    default: props => h('span', props.text)  },  // 如果子组件有定义 slot 的名称  slot: 'name-of-slot'  // 其他特殊顶层属性  key: 'myKey',  ref: 'myRef'}

vue-router2.x

1.router-view

在vue-router2中<router-view>是最顶层的出口,渲染最高级路由匹配到组件。同样地,一个被渲染组件同样可以包含自己的嵌套<router-view>

2.挂载

在router1.0中,挂载节点的方式为router.start()而在router2.0中使用vue自己的$mount手动挂载

3.给link添加事件

在vue-router1中使用v-link写入路由,但是在vue-router2中要使用router-link写入路由,在浏览器渲染的时候会把router-link渲染成a

有时候需要为router-link注册事件,对于一般的html元素,直接使用@click="eventFun"即可,但是对于router-link,像普通html元素那样注册事件后并不管用,需要添加.native才会成功注册

事实上给组件绑定原生事件就需要.native修饰v-on,否则无法注册成功。

<my-component v-on:click.native="doTheThing"></my-component>

4.利用vue-router做导航

在利用vue-router做导航的时候,需要用到redirect关键字的重定向功能,具体写法如下:

const router = new VueRouter({    routes : [        {path:'/',redirect:'/ZY'},        {path:'/ZY',component:ZY}    ]});

5.路由嵌套

vue-router的路由嵌套指的是子组件会在父组件中渲染出来,必须是子组件的父组件,祖先不可以实现,例如:

/user/foo/profile                     /user/foo/posts+------------------+                  +-----------------+| User             |                  | User            || +--------------+ |                  | +-------------+ || | Profile      | |  +------------>  | | Posts       | || |              | |                  | |             | || +--------------+ |                  | +-------------+ |+------------------+                  +-----------------+

原创粉丝点击