在线电影订票系统

来源:互联网 发布:js将时间戳格式化 编辑:程序博客网 时间:2024/04/29 02:23

在最近电影订票系统的实现中,使用了Vue.js作为前端开发框架。

介绍

Vue是一个极简的Javascript MVVM框架。它容易上手,便于整合。而且,它对于Web组件化开发的思考和设计,使其完全能够为复杂的SPA(单页应用)提供驱动。

这里写图片描述

与AngularJS1.x对比

在此之前,我主要都是使用AngularJs1.x作为前端框架进行开发。Angular是一个Javascript MVM框架,它也是Vue早起开发的灵感来源。但是,在我的使用过程中,发现Angular1.x还是存在着不少问题,而Vue就很好的解决了这些问题。个人认为主要问题有:

  1. API复杂。对于新手来说,需要投入大量的时间去熟悉并理解相关API。
  2. 不够灵活,模块化困难。在任何时候,都要遵循Angular制定的规则,这不是一个灵活的解决方案。
  3. 双向绑定。由于Angular使用双向绑定,导致应用中的数据流模糊复杂。在具体的实现过程中,常常会困在双向绑定的问题上。
  4. 指令与组件的区分不明显。Angular的指令中除了封装DOM操作,有时还会有视图和数据逻辑。但严格来说,指令应该只是封装DOM操作,组件才会有自己的视图和数据逻辑,并作为一个独立单元。
  5. 性能。Angular1.x的性能问题是被许多人诟病已久的了,也是我个人弃用Angular1.x的主要原因。在 Angular 1 中,当 watcher 越来越多时会变得越来越慢,因为作用域内的每一次变化,所有 watcher 都要重新计算。并且,如果一些 watcher 触发另一个更新,脏检查循环(digest cycle)可能要运行多次。Angular 用户常常要使用深奥的技术,以解决脏检查循环的问题。有时没有简单的办法来优化有大量 watcher 的作用域。

而以上问题,在Vue中已经得到了不错的解决,尤其是性能问题。Vue使用基于依赖追踪的观察系统并且异步队列更新,所有的数据变化都是独立触发,除非它们之间有明确的依赖关系。

MVVM

由于项目使用Vue框架,天然的就是一个MVVM模型,这里说明一下为什么不使用MVC模型以及为什么Vue是MVVM模型。

随着现代网页复杂度的增加,View 层所做的事,就不仅仅是简单的数据展示了,它不仅要管理复杂的数据状态,还要处理移动设备上各种操作行为等等。因此,前端也需要工程化,也需要一个类似于MVC 的框架来管理这些复杂的逻辑,使开发更加高效
一个标准的Web应用程序由这三部分组成:

View:UI布局,展示数据
Model:管理数据
Controller :响应用户操作,并将Model更新大片View上

这种 MVC 架构模式对于简单的应用来看起是可行的,也符合软件架构的分层思想。但实际上,随着H5 的不断发展,人们更希望使用H5 开发的应用能和Native 媲美,或者接近于原生App 的体验效果,于是前端应用的复杂程度已不同往日。这时前端开发就暴露出了三个痛点问题:

1.开发者在代码中大量调用相同的 DOM API, 处理繁琐 ,操作冗余,使得代码难以维护。
2.大量的DOM 操作使页面渲染性能降低,加载速度变慢,影响用户体验。
3.当 Model 频繁发生变化,开发者需要主动更新到View ;当用户的操作导致 Model 发生变化,开发者同样需要将变化的数据同步到Model 中,这样的工作不仅繁琐,而且很难维护复杂多变的数据状态。

于是MVVM模型诞生了,MVVM 由Model, View, ViewModel三部分构成

Model层代表数据模型,也可以在Model中定义数据修改和操作的业务逻辑;View 代表UI 组件,它负责将数据模型转化成UI 展现出来;ViewModel 是一个同步View 和 Model的对象。

在MVVM架构下,View 和 Model 之间并没有直接的联系,而是通过ViewModel进行交互,Model 和 ViewModel 之间的交互是双向的, 因此View 数据的变化会同步到Model中,而Model 数据的变化也会立即反应到View 上。

ViewModel 通过双向数据绑定把 View 层和 Model 层连接了起来,而View 和 Model 之间的同步工作完全是自动的,无需人为干涉,因此开发者只需关注业务逻辑,不需要手动操作DOM, 不需要关注数据状态的同步问题,复杂的数据状态维护完全由MVVM来统一管理。

既然项目使用Vue,那就说说其中MVVM实现的一些细节吧。
Vue.js 是采用 Object.defineProperty 的 getter 和 setter,并结合观察者模式来实现数据绑定的。当把一个普通 Javascript 对象传给 Vue 实例来作为它的 data 选项时,Vue 将遍历它的属性,用 Object.defineProperty 将它们转为 getter/setter。用户看不到 getter/setter,但是在内部它们让 Vue 追踪依赖,在属性被访问和修改时通知变化。

这里写图片描述

Observer 数据监听器,能够对数据对象的所有属性进行监听,如有变动可拿到最新值并通知订阅者,内部采用Object.defineProperty的getter和setter来实现。
Compile 指令解析器,它的作用对每个元素节点的指令进行扫描和解析,根据指令模板替换数据,以及绑定相应的更新函数。
Watcher 订阅者, 作为连接 Observer 和 Compile 的桥梁,能够订阅并收到每个属性变动的通知,执行指令绑定的相应回调函数。
Dep 消息订阅器,内部维护了一个数组,用来收集订阅者(Watcher),数据变动触发notify 函数,再调用订阅者的 update 方法。

从图中可以看出,当执行 new Vue() 时,Vue 就进入了初始化阶段,一方面Vue 会遍历 data 选项中的属性,并用 Object.defineProperty 将它们转为 getter/setter,实现数据变化监听功能;另一方面,Vue 的指令编译器Compile 对元素节点的指令进行扫描和解析,初始化视图,并订阅Watcher 来更新视图, 此时Wather 会将自己添加到消息订阅器中(Dep),初始化完毕。
当数据发生变化时,Observer 中的 setter 方法被触发,setter 会立即调用Dep.notify(),Dep 开始遍历所有的订阅者,并调用订阅者的 update 方法,订阅者收到通知后对视图进行相应的更新。

原创粉丝点击