vue2.x入坑总结—回顾对比angularJS/React

发布时间:2020-06-19 12:08:11 作者:周陆军
来源:网络 阅读:8673

从感性的角度讲,我是不屑于用VUE,觉得react套件用起来更顺手,但是vue现在越来火,所以也不得入vue(杂烩汤)的坑。vue/anguarJS/React,三者对关系现在就是:

vue2.x入坑总结—回顾对比angularJS/React

https://www.zhoulujun.cn/uploadfile/images/2018/0626/20180626214906428779269.jpg

自己ps了下,觉得深有道理,骚年们自己体悟,然后再问军哥^_^

不过回归真题,看vue还是先了解下https://cdn.zhoulujun.cn/vue.jpg(太大,自己打开)

vue生命周期及相关主题

组件实例周期

vue所有功能的实现都是围绕其生命周期进行的,在生命周期的不同阶段调用对应的钩子函数可以实现组件数据管理和DOM渲染两大重要功能。学习实例的生命周期,能帮助我们理解vue实例的运行机制,更好地利用钩子函数完成我们的业务代码。

create 和 mounted 相关 的函数有:

beforecreated》created》beforeMount》mounted》beforeDestroy

  1. created和beforeMount之间:首先会判断对象是否有el选项。如果有的话就继续向下编译,如果没有el选项,则停止编译,也就意味着停止了生命周期,直到在该vue实例上调用vm.$mount(el)。再次判断template参数选项的有无(因为vue需要通过el找到对应的outer template):

    (1)如果vue实例对象中有template参数选项,则将其作为模板编译成render函数。

    (2)如果没有template选项,则将外部HTML作为模板编译。

    (3)可以看到template中的模板优先级要高于outer HTML的优先级。

    如果没有template,则通过render传人的createElement编译

    综合排名优先级:render函数选项 > template选项 > outer HTML.

    这里没有看懂,可以看一下官方文档 独立构建和运行时构建

这这里顺便回顾下react生命周期

相比来讲,觉得react的生命周期更加清爽。建议参看:重谈react优势——react技术栈回顾

路由钩子

路由是项目等重点,很多事情可以在路由里面处理好。路由和store等规划项目基础架构核心,没有好的规划,工程就是一坨屎。

全局路由钩子

作用于所有路由切换,一般在main.js里面定义

组件路由勾子

和全局勾子不同的是,它仅仅作用于某个组件,一般在.vue文件中去定义。

beforeRouteEnter

这个是一个很不同的勾子。因为beforeRouterEnter在组件创建之前调用,所以它无法直接用this来访问组件实例。

为了弥补这一点,vue-router开发人员,给他的next方法加了特技,可以传一个回调,回调的第一个参数即是组件实例。

一般我们可以利用这点,对实例上的数据进行修改,调用实例上的方法。

我们可以在这个方法去请求数据,在数据获取到之后,再调用next就能保证你进页面的时候,数据已经获取到了。没错,这里next有阻塞的效果。你没调用的话,就会一直卡在那

beforeRouteLeave

在离开路由时调用。可以用this来访问组件实例。但是next中不能传回调。

beforeRouteUpdate

这个方法是vue-router2.2版本加上的。因为原来的版本中,如果一个在两个子路由之间跳转,是不触发beforeRouteLeave的。这会导致某些重置操作,没地方触发。在之前,我们都是用watch $route来hack的。但是通过这个勾子,我们有了更好的方式。


指令周期

bind:只调用一次,指令第一次绑定到元素时调用,用这个钩子函数可以定义一个在绑定时执行一次的初始化动作。

inserted:被绑定元素插入父节点时调用(父节点存在即可调用,不必存在于 document 中)。

实际上是插入vnode的时候调用。

update:被绑定元素所在的模板更新时调用,而不论绑定值是否变化。通过比较更新前后的绑定值,可以忽略不必要的模板更新。

慎用,如果在指令里绑定事件,并且用这个周期的,记得把事件注销

componentUpdated:被绑定元素所在模板完成一次更新周期时调用。

unbind:只调用一次, 指令与元素解绑时调用。

一个页面跳转,发生的事情

  1. 路由勾子 (beforeEach、beforeRouteEnter、afterEach)

  2. 根组件 (beforeCreate、created、beforeMount)

  3. 组件 (beforeCreate、created、beforeMount)

  4. 指令 (bind、inserted)

  5. 组件 mounted

  6. 根组件 mounted

  7. beforeRouteEnter的next的回调

  8. nextTick

结论:

路由勾子执行周期非常早,甚至在根实例的渲染之前

具体的顺序 router.beforeEach > beforeRouteEnter > router.afterEach

tip:在进行路由拦截的时候要避免使用实例内部的方法或属性

在开发项目时候,我们脑门一拍把,具体拦截的程序,写在了根实例的方法上了,到beforeEach去调用。结果导致整个拦截的周期,推迟到实例渲染的之后。

因此对于一些路由组件的beforeRouteEnter里的请求并无法拦截,页面看上去好像已经拦截下来了。

实际上请求依然发了出去,beforeRouteEnter内的函数依然执行了。

指令的绑定在组件mounted之前,组件的beforeMount之后

不得不提的, beforeRouteEnter的next勾子

beforeRouteEnter的执行顺序是如此靠前,而其中next的回调勾子的函数,执行则非常靠后,在mounted之后!!

我们通常是在beforeRouteEnter中加载一些首屏用数据,待数据收到后,再调用next勾子,通过回调的参数vm将数据绑定到实例上

因此,请注意next的勾子是非常靠后的。

nextTick:越早注册的nextTick触发越早


上文讲了这么多的router,顺势总结下:

router-link属性

:to :相当于a标签中的"herf"属性,后面跟跳转链接所用

replace:replace在routre-link标签中添加后,页面切换时不会留下历史记录

tag:具有tag属性的router-link会被渲染成相应的标签

active-class:这个属性是设置激活链接时class属性,也就是当前页面所有与当前地址所匹配的的链接都会被添加class属性

exact:开启router-link的严格模式

用了vue-router,证明项目工程还是蛮大的,建议使用vuex来做全局数据管理(可能用redux习惯了吧!)

Vuex下Store的模块化拆

vuex的store天生自带modules概念,同时也需要thunk中间件,action处理异步数据。下面copy一点我项目code demo:

const store = new Vuex.Store({
  modules: {
    authForm: authFormStore,
    BankAdd:BankAddFormStore
  }
});

下面是store的示范

const authFormStore = {
  state: {
    name: ""
  },
  mutations: {
    update: function (state, obj) { //我原来也是写了N多个update函数,图样图森破啊
      state[obj.name] = obj.value;
    },
  },
  action: {
    setData: function (context, obj) {
      //TODO
      Vue.http.post("api", params, {emulateJSON: true}).then(function (res) {
        // 处理业务
        // 调用上面setAgree方法更新点赞数
        context.commit(obj.name, obj.value);
      }, function () {
      })
    },
  },
  getters: {
    getNews(state){
      //TODO  return new value
    }
  }
}

在vue组件mapState

computed: {
  memberType: function () {
    let memberType = this.$route.params.memberType;
    this.initWeChat(window.location.href, memberType);
    return memberType;
  },
  ...mapState({
    authForm: state => state.authForm,
    epCertType: state => state.authForm.epCertType
  })
},

这里有坑:表单的v-model属性值是Vuex的state时,如果时严格模式,因为用户输入时,v-model会试图修改v-model的值,由于修改并非mutation执行的,严格模式下会抛出错误。

针对这种情况,有两个处理方法:一个是双向绑定的计算属性,一个是给表单绑定value,然后侦听input或change事件,在事件中调用action。

computed:{
    message:{
      get(){
        return this.$store.obj.message
      },
      set(value){
        this.$store.commit('updateMessage',value)
      }
    }
}

第二种方法

computed:{
 ...mapState({
  message: state => state.obj.message
 })
},
methods:{
 updateMessage(e){
  this.$store.commit('updateMessage',e.target.value)
 }
}
mutations:{ //store mutation函数
 updateMessage(state,message){
  state.obj.message=message
 }
}


 Vue.js 最核心的功能有两个,一是响应式的数据绑定系统,二是组件系统

所谓双向绑定,指的是vue实例中的data与其渲染的DOM元素的内容保持一致,无论谁被改变,另一方会相应的更新为相同的数据。这是通过设置属性访问器实现的。

关于vue\angluar\react的数据绑定:双向绑定和单向数据流

Vue 的依赖追踪是【原理上不支持双向绑定,v-model 只是通过监听 DOM 事件实现的语法糖】

vue的依赖追踪是通过 Object.defineProperty 把data对象的属性全部转为 getter/setter来实现的;当改变数据的某个属性值时,会触发set函数,获取该属性值的时候会触发get函数,通过这个特性来实现改变数据时改变视图;也就是说只有当数据改变时才会触发视图的改变,反过来在操作视图时,只能通过DOM事件来改变数据,再由此来改变视图,以此来实现双向绑定

双向绑定是在同一个组件内,将数据和视图绑定起来,和父子组件之间的通信并无什么关联

组件之间的通信采用单向数据流是为了组件间更好的解耦,在开发中可能有多个子组件依赖于父组件的某个数据,假如子组件可以修改父组件数据的话,一个子组件变化会引发所有依赖这个数据的子组件发生变化,所以vue不推荐子组件修改父组件的数据,直接修改props会抛出警告

这里推荐阅读《Vue.js双向绑定的实现原理》

react没有数据双向绑定

react是单向数据流:对应任何可变数据理应只有一个单一“数据源”,数据源状态提升至父组件中

react中通过将state(Model层)与View层数据进行双向绑定达数据的实时更新变化,具体来说就是在View层直接写JS代码Model层中的数据拿过来渲染,一旦像表单操作、触发事件、ajax请求等触发数据变化,则进行双同步

angular也是双向数据绑定(一次做完所有数据变更,然后整体应用到界面上)

安利下:再谈angularJS数据绑定机制及背后原理—angularJS常见问题总结

三者中,我还是更推崇react+redux模式(自上而下的数据流,业务聚焦于数据树设计)

这里面不得不提的就是,vuejs对data中数组的原生方法进行了封装,所以在改变数组时能够触发视图更新。这个我在写日期控件对时候遇到很多坑,比如:

  1. 通过索引直接修改数组的元素,例如vm.items[0] = {title: 'title'}

  2. 无法直接修改数组的长度,例如vm.items.length = 0

解决方案: 对于第一种vue提供了set方法vm.items.set(0,{title: ‘title’}) 或vm.$set(‘items[0]’,{title: ‘title’})。另外一个列表渲染的时候的有一个性能的小技巧: 如果数组中本身自带一个唯一的标识 id ,那么在渲染的时候,通过trace-by给数组设定唯一的标识,这样vuejs在渲染过程中会尽量重复原有对象的作用域和dom元素。


关于用“箭头函数精简你的 Vue 模块”(建议点击阅读),精简出来就是:

methods: {
  method(){//todo}  //不用: method:()=>{//todo}   method:function(){//todo}
}


参考文章:

Vue2.0 探索之路——生命周期和钩子函数的一些理解

详解vue生命周期

vue生命周期

vue生命周期探究

vue安装,router-link的一些属性,用法,tag active-class,to,replace,exex等等

Vuex下Store的模块化拆分实践

Vue.js与React的全面对比

Vue.js 2.0源码解析之前端渲染篇

用箭头函数精简你的 Vue 模块 – dotdev

转载请注明来源,vue2.x入坑总结-回顾对比angularJS/React - vue入坑总结 - 周陆军的个人网站,:https://www.zhoulujun.cn/html/webfront/ECMAScript/vue/8125.html。不妥之处,望告之,谢谢!


推荐阅读:
  1. Vue 中scoped CSS 与深度作用选择器 /deep/
  2. vue.js+vscode+visual studio在windows下搭建开发环境

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

vue vuex react

上一篇:【Axure原型图】—— 4. Tab Control(选项卡)

下一篇:保证 Linux 服务器安全基本措施

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》