引子:
最近工做挺忙,avalon只能断断续续的写下去了,大概看了下angular的源码,看到小一半就比较难坚持了,是块硬骨头,慢慢啃吧html
不过angular的的文档中用词仍是很优雅:json
- HTML编译器
- 指令
- 编译
- 连接
- 过滤器
- 注入器
- 控制器
- 管道
等等…看起来以为老高级,其实avalon也间接的部分实现,原理也不是很复杂数组
avalon版本更新很快,新版的加入了AMD规范的模块加载器,还修复了不少BUG,不过相信短时间内实现的核心仍是不会变化,因此我依然以如今的版本分析为主并发
编译期
- 视图背后的代码就是控制器,在mvvm中就是vm视图模型,它的主要工做就是构造模型,并把模型与回调方法一并发送到视图,视图能够看做做用到模版HTML上的投影
- 因此在编译阶段,咱们的控制器就会把用户定义的数据模型给构造出来
- avalon中的modelFactory工厂方法构造出的model对象其实就是真正的控制器了,至于构造出来的控制器如何注入到视图上的,等之后分析到HTML编译器双向绑定吧
- model 原本是系统内部定义的一个临时对象,将控制器和avalon的做用域对象给关联起来
接上一节app
收集用户定义的scope在过滤的时候作了2个处理mvvm
callGetters.push(accessor);
callSetters.push(name);
收集控属性赋监与计算属性,是为了在初始化scpoe中的代码未处理的方法函数
处理监控属性
//给控属性赋监值,调用对应监控属性的set ->accessor方法
callSetters.forEach(function(prop) {
// model.firstName = '司徒' ->调用了 model.firstName->set->accessor方法
model[prop] = scope[prop]; //为空对象赋值
});
遍历监控属性收集器,给初始化的空model对应的方法赋值this
这里注意各重点,赋值的的时候实际是调用的accessor方法,由于set get给转换过了spa
accessor 源码prototype
accessor = function(neo) { //建立监控属性或数组
//若是有参数
if (arguments.length) {
//用于改变用户定义的函数内部能访问正确的是vm模型时,会触发这个处理须要跳过
if (stopRepeatAssign) {
return; //阻止重复赋值
}
if (value !== neo) {//传的值与旧的不相等
var old = value;
//监控数组:定义时为一个数组
if (valueType === "Array" || valueType === "Object") {
if (value && value.$id) {
updateViewModel(value, neo, Array.isArray(neo));
} else if (Array.isArray(neo)) {
value = Collection(neo, model, name);
} else {
value = modelFactory(neo);
}
} else {
//若是是简单类型
value = neo;
}
//每次更新都会修正为新的赋值
json[name] = value && value.$id ? value.$json : value;
//更新依赖,就是当前的操做会触发与之相关
notifySubscribers(accessor); //通知顶层改变
model.$events && model.$fire(name, value, old);
}
} else {
collectSubscribers(accessor); //收集视图函数
return value;
}
};
其实源码注释很清楚了,咱们概括下执行的流程
- 判断参数是调用set仍是get方法
- stopRepeatAssign //阻止重复赋值,是这factory.apply(0, deps);重置上下文的时候处理的
- 监控数组处理
- 更新json //收集原始的定义
- notifySubscribers //更新依赖,就是当前的操做会触发与之相关
- model.$events 触发订阅的自定义事件
notifySubscribers 其实就是关键的执行点,执行当前做用域所依赖的全部的,这个在双向绑定的时候就能够仔细讨论了
处理计算属性:

监控属性涉及用户定义的处理,因此要作不少关联的处理
流程:
- 收集依赖关系
- 处理用于定义的get方法
- 更新json
- 返回定义函数的结果
Publish 对象是将函数曝光到此对象上,方便访问器收集依赖
fn.nick 就是对应的计算属性方法名称,在过滤的时候 accessor.nick = name;附上的
一样执行了accessor方法,因为没有传递参数,实际上就是在处理收集依赖关系了
accessor 源码
accessor = function(neo) { //建立计算属性
//@第三层做用域
if (arguments.length) {
if (stopRepeatAssign) {
return; //阻止重复赋值
}
if (typeof setter === "function") {
setter.call(model, neo);
}
if (oldArgs !== neo) { //因为VBS对象不能用Object.prototype.toString来断定类型,咱们就不作严密的检测
oldArgs = neo;
notifySubscribers(accessor); //通知顶层改变
model.$events && model.$fire(name, neo, value);
}
} else {
if (openComputedCollect || !accessor.locked) {
collectSubscribers(accessor);
}
//解析出get函数,返回新的值
return value = json[name] = getter.call(model); //保存新值到json[name]
}
};
collectSubscribers方法

很明显的处理,取出开始push到的Publish的处理回调,取出依赖列表,合并
ensure 法只有当前数组不存在此元素时只添加它

全部此时的 subscibers关联就有值了
最后执行定义的get方法,更新json

注意的一点

这里又涉及到取值的问题,因此又会关对应的执行各自的accessor
因此这里会进行一次收集依赖了
在转换的完毕model后,会给model增长订阅的特性与一些属性
- model.$json = json; //纯净的js对象,全部访问器与viewModel特有的方法属性都去掉
增长事件订阅
- model.$events = {}; //VB对象的方法里的this并不指向自身,须要使用bind处理一下
- model.$watch = Observable.$watch.bind(model);//用于监听ViewModel中的某属性变化,它将新值与旧值都传给回调
- model.$unwatch = Observable.$unwatch.bind(model);//卸载$watch绑定的回调
- model.$fire = Observable.$fire.bind(model); //触发$watch指定的回调
ViewModel的ID,方便经过avalon.models[$id]访问
- model.$id = generateID();
判断是否为模型中的原始数据
最后返回工厂转化后的model对象
主方法入口
avalon.define

其实这里有一种重点

做者再次把定义的模型给执行了一遍,用意呢?
请看
vm.xxx = 1;
vm.fullName = fucntion(){
vm.xxxx
}
在定义的VM中的方法中,若是再次访问vm.xxx属性,
这时候内部引用不对了 VM仍是指向原来的普通JS对象,而不是真正的VM因此须要apply一次,改变
那么有个精妙的思路:
咱们 factory.apply(0, deps); //重置它的上下文
因此把方法执行一次把内部引用换给model
由于转换了模型关系,因此监控属性与计算属性都会有对应的set get操做了,相对应的上下文也变成了vm了
stopRepeatAssign return 阻止了,防止重复赋值
avalon.models[name] = model; 挂到了全局的models中,方面之后使用
下章就开始讲HTML编译器与指令,如何真正开始工做了