Angular 和 Vue

在对Angular的学习中,了解到AngularJS 的两个主要缺点:

  • 对于每一次界面时间,Ajax 或者 timeout,都会进行一个脏检查,而每一次脏检查又会在内部循环检查,当界面绑定的model 很多,就会造成严重的性能问题 。
  • Angular 混乱的模块,并不能起到命名空间的作用,因而在大项目中组织模块又是一件头疼的事。

但是 Angular 开发中小型的应用是很棒的,也是将 MVC 引入到前端的开始。定一个目标,两年时间理解 《Build Your Own AngularJS》。

Vue 抛弃了Angular 中脏检查的方式,而使用 Javascript 中变量的setter 属性来截获数据变化,更加巧妙和机智。

在 MVVM 时代,浏览器自身属性变得更加重要起来。getter 和 setter 曾被认为是一个无太大用处的一个属性。

为了较为深入的理解 Vue, 自己撸了一个简易版 MVVM 。 先看效果吧 。

    <input type="text" v-model="hello" value="">
<div id="template">
{{hello}}
</div>
<div class="template">
{{bind}}
</div>
<script src="app.js"></script>
    var vue = new Vue({
ele: '#template',
});
var vue1 = new Vue({
ele: '.template',
data: {
bind: 'hey i am a vue lover'
}
})

开撸开撸

若有关于 gettersetter 问题 ,请先移步 MDN 。

申明Vue 对象

function Vue(option) {
this.init(option);
}

初始化

Vue.prototype.init = function(option) {
this._data = option.data || {};
this._method = option.method || {};
this.bingdings = {};
this.elements = typeof option.ele === "string" ? document.querySelectorAll(option.ele) : option.ele;
this.bind();
this.observe();
this.react();
this.initMvvM();
}

将界面上的模型数据绑定到后台


Vue.prototype.bind = function() {
for (var i = 0, length = this.elements.length; i < length; i++) {
var ele = this.elements[i],
html = ele.innerHTML,
spans, span, dataAttr;
html = html.replace(/\{\{(.*?)\}\}/g, function(a, b) {
var span = '<span v-data="' + b + '"></span>';
return span;
})
ele.innerHTML = html;
spans = ele.querySelectorAll('[v-data]');
for (var j = 0, l = spans.length; j < l; j++) {
span = spans[j];
dataAttr = span.getAttribute('v-data');
if (!this.bingdings[dataAttr]) {
this.bingdings[dataAttr] = { value: this._data[dataAttr] || '', ele: [] };
} this.bingdings[dataAttr].ele.push(span);
span.innerHTML = this.bingdings[dataAttr].value;
span.removeAttribute('v-data');
}
}
}

// 核心部分,使用 setter 观测 _data

Vue.prototype.observe = function() {
var self = this,
eles;
for (var key in self.bingdings) {
Object.defineProperty(self._data, key, {
get: function() {
return self.bingdings[key].value;
}, set: function(newVal) {
if (newVal !== self.bingdings[key].value) {
self.bingdings[key].value = newVal;
eles = self.bingdings[key].ele;
for (var i = 0, l = eles.length; i < l; i++) {
eles[i].innerHTML = newVal;
}
}
}
})
}
}

到这里,_data_ 的赋值已经可以引起界面上dom 的变化。也就是说完成了 model 到view 的数据绑定和响应。

现在来完成 dom 到 view 的事件响应。


Vue.prototype.initMvvM = function() {
var self = this;
var models = document.querySelectorAll('[v-model]'),
model, dataModel;
for (var i = 0, l = models.length; i < l; i++) {
model = models[i];
dataModel = model.getAttribute('v-model');
self._data[dataModel] = model.value;
}
}

// 事件响应

Vue.prototype.react = function() {
var self = this;
var models = document.querySelectorAll('[v-model]'),
model, dataModel;
for (var i = 0, l = models.length; i < l; i++) {
model = models[i];
model.addEventListener('change', function(e) {
dataModel = this.getAttribute('v-model');
self._data[dataModel] = this.value;
}) model.addEventListener('keyup', function(e) {
dataModel = this.getAttribute('v-model');
self._data[dataModel] = this.value;
})
}
}

最后,更新init 函数。

Vue.prototype.init = function(option) {
this._data = option.data || {};
this._method = option.method;
this.bingdings = {};
this.elements = typeof option.ele === "string" ? document.querySelectorAll(option.ele) : option.ele;
this.bind();
this.observe();
this.react();
this.initMvvM();
}

总结

大功告成。

但是相比与真正的Vue还是差了很多,比如 这里的数据绑定仅支持基本类型,函数绑定也没有完成。在Vue 中,每一个监控的属性都会设置依赖,从而陷入避免和 Angular 中一样的循环检查。

另外,在Vue中,数据模版并不是这样使用插入一个span,而是创建一个 文本节点,然后append 到当前父元素上。

这只是一个简易版的,把所有的部分都放在Vue上,非常不合适。 后面会进行重构,把数据监控使用订阅和发布模式封装。

在这之前,需要进行模块化和依赖注入,所以,接下来要完成一个简易的requirejs,会提供三个api ,define, require 和 use。

代码托管到了 github(但是今天的网速不好,改天补充.),

如有错误,希望不吝赐教。

如果您有合适的前端工资机会,也期待您的邮件。

基于 getter 和 setter 撸一个简易的MVVM的更多相关文章

  1. 基于Win32 SDK实现的一个简易线程池

    利用C++实现了一个简易的线程池模型(基于Win32 SDK),方便使用多线程处理任务.共包含Thread.h.Thread.cpp.ThreadPool.h.ThreadPool.cpp四个源文件. ...

  2. 手把手教你撸一个简易的 webpack

    背景 随着前端复杂度的不断提升,诞生出很多打包工具,比如最先的grunt,gulp.到后来的webpack和Parcel.但是目前很多脚手架工具,比如vue-cli已经帮我们集成了一些构建工具的使用. ...

  3. 曹工说Tomcat2:自己撸一个简易Tomcat Digester

    一.前言 框架代码其实也没那么难,大家不要看着源码就害怕,现在去看 Tomcat 3.0的代码,保证还是看得懂一半,照着撸一遍基本上很多问题都能搞定了.这次我们就模拟 Tomcat 中的 Digest ...

  4. 半天撸一个简易版mybatis

    为什么需要持久层框架? 首先我们先看看使用原生jdbc存在的问题? public static void main(String[] args) { Connection connection = n ...

  5. 撸一个简单的MVVM例子

    我个人以为mvvm框架里面最重要的一点就是VM这部分,它要与Model层建立联系,将Model层转换成可以被View层识别的数据结构:其次也要同View建立联系,将数据及时更新到View层上,并且响应 ...

  6. C#基于Mongo的官方驱动手撸一个Super简易版MongoDB-ORM框架

    C#基于Mongo的官方驱动手撸一个简易版MongoDB-ORM框架 如题,在GitHub上找了一圈想找一个MongoDB的的ORM框架,未偿所愿,就去翻了翻官网(https://docs.mongo ...

  7. 撸了一个简易的配置中心,顺带整合到了SpringCloud

    大家好,我是三友~~ 最近突然心血来潮(就是闲的)就想着撸一个简单的配置中心,顺便也照葫芦画瓢给整合到SpringCloud. 本文大纲 配置中心的概述 随着历史的车轮不断的前进,技术不断的进步,单体 ...

  8. 手撸一个springsecurity,了解一下security原理

    手撸一个springsecurity,了解一下security原理 转载自:www.javaman.cn 手撸一个springsecurity,了解一下security原理 今天手撸一个简易版本的sp ...

  9. 基于注解处理器开发自动生成getter和setter方法的插件

    昨天无意中,逛到了lombok的网站,并看到了首页的5分钟视频,视频中的作者只是在实体类中写了几个字段,就可以自动编译为含setter.getter.toString()等方法的class文件.看着挺 ...

随机推荐

  1. CoreCLR源码探索(一) Object是什么

    .Net程序员们每天都在和Object在打交道 如果你问一个.Net程序员什么是Object,他可能会信誓旦旦的告诉你"Object还不简单吗,就是所有类型的基类" 这个答案是对的 ...

  2. C++中的const

    一,C++中const的基本知识 1.C++中const的基本概念 1.const是定义常量的关键字,表示只读,不可以修改. 2.const在定义常量的时候必须要初始化,否则报错,因为常量无法修改,只 ...

  3. app引导页(背景图片切换加各个页面动画效果)

    前言:不知不觉中又加班到了10点半,整个启动页面做了一天多的时间,一共有三个页面,每个页面都有动画效果,动画效果调试起来麻烦,既要跟ios统一,又要匹配各种不同的手机,然后产品经理还有可能在中途改需求 ...

  4. 计算机程序的思维逻辑 (60) - 随机读写文件及其应用 - 实现一个简单的KV数据库

    57节介绍了字节流, 58节介绍了字符流,它们都是以流的方式读写文件,流的方式有几个限制: 要么读,要么写,不能同时读和写 不能随机读写,只能从头读到尾,且不能重复读,虽然通过缓冲可以实现部分重读,但 ...

  5. 【转】39个让你受益的HTML5教程

    闲话少说,本文作者为大家收集了网上学习HTML5的资源,期望它们可以帮助大家更好地学习HTML5. 好人啊! 不过,作者原来说的40个只有39个,因为第5个和第8个是重复的. 原文在此! 1. 五分钟 ...

  6. ActionContext.getContext().getSession()

    ActionContext.getContext().getSession() 获取的是session,然后用put存入相应的值,只要在session有效状态下,这个值一直可用 ActionConte ...

  7. jmeter之线程组的使用

    线程组 在使用jmeter性能测试时,我们都得先添加个线程组,右键testplan-->添加-->Threads-->线程组.在线程组下执行. 问题:为了能够让jmeter在做性能测 ...

  8. python selenium

    https://segmentfault.com/a/1190000007249396?_ea=1293878

  9. 如何利用mono把.net windows service程序迁移到linux上

    How to migrate a .NET Windows Service application to Linux using mono? 写在最前:之所以用要把windows程序迁移到Linux上 ...

  10. 提升网速的路由器优化方法(UPnP、QoS、MTU、交换机模式、无线中继)

    在上一篇<为什么房间的 Wi-Fi 信号这么差>中,猫哥从微波炉.相对论.人存原理出发,介绍了影响 Wi-Fi 信号强弱的几大因素,接下来猫哥再给大家介绍几种不用升级带宽套餐也能提升网速的 ...