vue与react类(对)比学习总结

用了一年半的react系,由于新公司项目都是vue系的,所以开始学习vue相关框架。早就听说vue是angular和react的结合,angular没用过,react和大牛们一起做产品该踩的坑坑也基本都踩完了,本着触类旁通的理念,在学习vue的时候有意识的和react进行类(对)比,加上近期在vue-electron项目上练了练手,做下总结。

一、 生命周期

vue (react)

  • beforeCreate
  • created
  • beforeMount (componentDidMount)
  • mounted (componentDidMount)
  • beforeUpdate (componentWillReceiveProps, shouldComponentUpdate, componentWillUpdate)
  • updated (componentDidUpdate)
  • beforeDestroy (componentWillUnmount)
  • destroyed (componentDidUnmount)

括号中是react相关的生命周期,可以看出从beforeMount开始和react基本一致,只是react在视图更新前做了更细粒度的控制,componentWillReceiveProps, shouldComponentUpdate, componentWillUpdate 这三个方法都接收nextProps和nextState两个参数,允许你在这三个方法中针对新数据决定是否渲染,以及如何渲染的控制。

vue中beforeCreatecreated这两个方法是react中没有的,这也是vue与react本质不同的体现,vue中有data、computed、method等等属性,这两个方法就是把这些属性注入到模板中的过程,注入后才参与后面模板的编译,生成dom,挂载等等操作。在beforeCreate中访问实例的data这些属性是undefined,所以在这里可以用于设置data的初始值,created中这些属性就已经被绑在实例上可以访问了。

二、vue实例属性方法

data

vue中的data属性类似于react中的state,属于组件的私有数据,不同在于react中state是通过setState()方法进行更新的,而vue中data除了通过直接赋值改变其值,还可以通过v-model这个语法糖进行数据双向绑定,这个主要用于表单组件。

props

vue中props和react中的props类似,不同的是vue中的props来源于父组件,而react中props除来自于父组件还来自于redux中的数据。

computed

computed属性是vue的一大特色,它依赖于props、data生成新的计算数据,会根据props和data的改变而改变。computed中的数据默认只有getter属性,也可设置setter属性。在react中想对props或者state再加工一般都是另外再初始化一个state数据或者在render函数中创建临时变量,特别是一些在显示前的排序过滤方法都在render里调用。另外computed中还包括vuex中mapState和mapGetters的数据。

method

method是放各种方法的地方,包括vuex中mapActions的方法,而react中,react-redux将action取出来作为props传入组件中。

watch

watch一般用于观测数据变化不会触发视图更新的数据,或者是需要根据数据变化做其他操作情景,比如子组件数据更新通知父组件。watch的回调函数接收(val, oldVal),这个和react中更新前的几个生命周期的用法有点类似,props或者state改变就会触发。而且watch的数据有deep配置项,可以监测对象内部的变化。

三、vue组件

vue和react都是组件化、模块化思想的产出,vue推荐单文件组件,就是把template、js、css写在一个文件里,这个优点就是修改代码不用切换文件,每个组件的样式、模板、事件都在一个文件里,react虽然是JSX,但是样式控制在单独的文件里(现在也有了CSS in JavaScript的样式框架)。不过vue这样有个缺点就是组件稍微大一点,单个文件代码行数就暴多,轻而易举就上300行。

组件通信

vue的组件通信有三种方法,下面依次讲讲:

  1. props & $on & $emit
    这种方法主要用于父子组件间的通信,父组件通过绑定动态props将父数据传给子组件,子组件通过$emit事件触发父组件上通过$on绑定的自定义事件,并将子组件的数据抛上去。
  2. eventBus & $emit
    eventBus常用于同代组件间的通信,它是实际上是另外定义的一个全局vue实例,在此实例上订阅t和触发事件可以在全局有效。使用方法和前一种一样,但是这个在组件中订阅时要记得卸载前取消事件订阅,因为vue不会在生命周期结束时自动取消eventBus事件的订阅,会造成全局污染。而且这种方法适用于单个组件,如果是list那种循环组件,不宜在每个list item上进行全局事件订阅。
    按理说在有了vuex不需要eventBus,但在此个人认为,vuex是管理数据状态的,是一个全局数据存储的地方,如果不是全局都要用的,不跨路由,就不需要存起来,应该考虑用eventBus,不能说有了vuex,props什么的都不用了,什么数据都丢到vuex,要用的都从里面取,随着项目复杂度的增加,数据管理就会变得杂乱,我们应该像后端设计数据库一样来设计前端的数据管理。
  3. vuex
    上一种方法中也说到了vuex,vuex其实能cover上面两种方法的情景,但是就像上面说的,不能说vuex好用就滥用,就像好吃的东西吃多也会撑死。根据上一个react-redux产品项目中redux的应用,我们就是将全局需要的数据才存到redux,供跨路由数据存取,不需要的都通过props,state来解决,就是这样,随着产品迭代,新功能增加,store中的数据都变得难以维护,再加上最开始数据结构设计过于依赖业务逻辑,导致一些扩展困难。所以前端数据结构设计也很重要。
    vuex还是应该用于存储全局数据,跨路由,跨多个组件,比如祖父子组件间的通信,业务之间跨越比较大的组件间的通信。

四、vuex

vuex是专为vue设计的数据管理,就vuex和redux的异同我做了些研究,他们本质区别是redux设定数据是immutable的, 而vuex设定数据是mutation的,redux中每次数据更新都会生成新的store tree,而vuex就是在store tree上直接修改。下面针对vuex各个部分和redux对比总结下。

state

state就是store中的数据,是唯一数据源的存在,这个概念和redux中store是一样的,通过辅助方法mapState,可将多个state解构在组件computed方法中,这个和react-redux中的mapStateToProps类似。

getter

getter是对state的再加工,类似于vuex中的计算属性,他和react项目中reselect插件实现一样的功能,vuex将其集成了。通过mapGetters可以将这些getter映射到局部计算属性,react中则是直接import。

mutation

mutation类似redux中reducer的作用,数据的改变就发生在这个部分,这个部分也是vuex和redux本质区别的地方,vuex在mutation中改变store上的相关数据的值,而redux在这里是根据actiontype和playload返回的是一个新的数据。另外mutation中是同步函数,异步函数不能在此执行。

action

action和redux中的action类似,可以执行异步操作,可以分发action,和redux中的middleware类似。

modules

modules将store分割成不同模块,每个模块拥有自己的state、mutation、action、getter,这个思想和redux中reducer拆分类似,不过reducer合并的时候生成了新的store tree。

vuex的store中的状态是响应式的,和vue遵循一样规则:

  1. 初始化所有的所需属性
  2. 当需要再对象上添加新属性是,应该用Vue.set或者用对象展开运算符

这个规则是我觉得vue挺尴尬的地方,他无法监测到对象内部的变化,你直接赋值改变对象他是无感的,无法响应,经常出现数据已经更改,视图没有更新。对于非引用类型的数据他让你直接赋值更改,然后响应,也就是mutation,而对于引用类型的数据,你必须给以新对象替换老对象它才能响应,个人觉得这是一种不一致性的表现。虽然它给出了Vue.set的解决方法,但是在2.0中,Vue.set不接受keyPath的方式动态增加属性,很多时候存在超过二级嵌套属性时,就贼尴尬了。在社区里找了半天,找到了vue-deepset插件,然后就完美解决vue和vuex中动态增加属性的问题。

Personal opinion

由于我刚入前端时就学习的是react框架,没什么历史负担,所以对于react具有侵略性的JSX是很快就接受了,它将JS和HTML部分糅合在一起了,这对于老前端来说是很难接受的,但是前端组件化、模块化是大势所趋,vue的出现解决了这个问题,记得vue的英文版是这么描述自己的

Vue (pronounced /vjuː/, like view) is a progressive framework for building user interfaces. Unlike other monolithic frameworks, Vue is designed from the ground up to be incrementally adoptable.

progresive渐进式增强,我理解的另一个意思,就是一步一步将以前的项目向组件化、模块化演变提供了循序渐进的途径。

从vue2.0开始,引入了render,虚拟dom,这些都是在向react靠齐,虽然没有用过angular,但是感觉vue更像react。关于template的概念在用jekyll搭建博客的时候就有涉及,所以我想这必定不是angular的特色。如果说vue相对于react的特色那就是它的各种指令,非常直接方便灵活,比react那套props,state,method严格的单向数据流要灵活许多。

除了上面提到的vue处理数据mutate的不一致性,另一个让我感觉不舒服的地方就是在非字符串模式时,template中要求kebab-case,这是HTML模板大小写不敏感限制的,在组件声明的时候又是PascalCase,dev-tool中又将模板中的kebab-case自动转成了PascalCase,对于我初学调试的时候甚是费解,如果组件文件命名再不一致,根本找不到调试的组件在哪,后来vue的dev-tool升级允许查看单个组件的kebab-case形式,这些不一致和react中的JSX比起来就显得尴尬了。

总的来说,我还是对react比较有好感,因为它很严格,一致性很好,vuex就要花哨灵活了,对于复杂大型的应用还是react更好些。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容

  • Vue也已经升级到2.0版本了,到现在为止(2016/11/19)比较流行的MVVM框架有AngularJS(也有...
    彬_仔阅读 27,217评论 12 114
  • 系列文章:Vue 2.0 升(cai)级(keng)之旅Vuex — The core of Vue applic...
    6ed7563919d4阅读 4,547评论 2 58
  • 你问我恨不恨你,如果你跟小明在一起的话。一遍又一遍,我说不恨。你问为什么,开始我说一些冠冕堂皇的心灵鸡汤安慰自己...
    徐小点阅读 441评论 0 0
  • 感觉自从今天开始上班之后,整个人立马变得精神抖擞生龙活虎起来,仿若久旱逢甘霖般的复苏了。原来,工作才是治愈无聊和抑...
    朝凪阅读 182评论 0 0
  • 2018年开篇还是在告别过去,自己总是慢半拍, 好在「Better late than never」, 一声迟到的...
    窦子V阅读 553评论 2 1