javascript ORM

前端ORM框架其实也就是一个对于DAO数据访问接口的封装,主要是封装CRUD四种类型基本操作。
所谓对象关系映射的构建,最基本的还是在于模型这一层,也就是数据模型,我们应该用对象来封装我们的数据,以形成模型。
例如一个基本的数据结构为:

1 var data = {
2     name: 'ken',
3     age: 18
4 };

我们需要将这样一个数据转化为一个对象,可以将之作为一个参数传递到一个具有某些功能的对象中去,例如:

1 function Model(data) {
 2     this.data = data;
 3 }
 4 Model.prototype.save = function() {
 5     //do create or update
 6 };
 7 Model.prototype.read = function() {
 8     
 9 };
10 Model.prototype.delete = function() {
11     
12 };
13 Model.prototype.get = function(attr) {
14     return this.data[attr];
15 };

上面的代码定义了一个基本的数据模型,这个能够完成与服务器之间的数据交互,而数据最终还是需要展现在页面上的,这就需要另外一个对于数据和视图之间的处理了。数据变化之后能够通知页面进行重新渲染,我们可以使用比较常用的一种设计模式,也就是发布/订阅模式来完成这种事件的交互。在数据层发布事件,在视图层去监听事件。我们可以在网上找到很多实现了该功能的代码,其原理也很简单,只需要维护一个用于保存事件名和相应事件的Map表即可,下面是一个实现代码:

1 var Events = Spine.Events = {
 2     bind: function(ev, callback) {
 3       var evs   = ev.split(" ");
 4       var calls = this._callbacks || (this._callbacks = {});
 5       
 6       for (var i=0; i < evs.length; i++)
 7         (this._callbacks[evs[i]] || (this._callbacks[evs[i]] = [])).push(callback);
 8 
 9       return this;
10     },
11 
12     trigger: function() {
13       var args = makeArray(arguments);
14       var ev   = args.shift();
15             
16       var list, calls, i, l;
17       if (!(calls = this._callbacks)) return this;
18       if (!(list  = this._callbacks[ev])) return this;
19       
20       for (i = 0, l = list.length; i < l; i++)
21         if (list[i].apply(this, args) === false)
22           return false;
23       return this;
24     },
25 
26     unbind: function(ev, callback){
27       if ( !ev ) {
28         this._callbacks = {};
29         return this;
30       }
31       
32       var list, calls, i, l;
33       if (!(calls = this._callbacks)) return this;
34 
35       if (!(list  = this._callbacks[ev])) return this;
36       
37       if (!callback) {
38         delete this._callbacks[ev];
39       }
40       
41       for (i = 0, l = list.length; i < l; i++) {
42         if (callback === list[i]) {
43           list.splice(i, 1);
44           break;
45         }
46       }
47         
48       return this;
49     }
50   };

我们可以给任何需要用到发布订阅模式的对象继承该Event对象,然后我们便可以很方便的发布/订阅事件了。
当然,我们的对象默认情况下并没有一个可直接使用的继承方法,但是要实现基层并不难,我们可以给我们的Model对象添加下面这两个方法:

1 Model.include = function(obj){
 2   for(var key in obj) {
 3       this.prototype[key] = obj[key];
 4   }
 5 
 6   var included = obj.included;
 7   if (included) included.apply(this);
 8   return this;
 9 },
10 
11 Model.extend = function(obj){
12   for(var key in obj) {
13       this[key] = obj[key];
14   }
15   
16   var extended = obj.extended;
17   if (extended) extended.apply(this);
18   return this;
19 }

有了上面的两个方法之后,我们便可以通过include方法来扩展实例方法,通过extend来扩展静态方法,对于Event,我们可以用extend来进行扩展:

1 Model.extend(Event);

扩展完成之后,我们便可以在我们的代码中添加事件的发布或者订阅了,在模型中有关于数据保存或者更新的地方我们可以发布change事件,在数据销毁的时候我们可以发布destroy事件:

1 Model.prototype.save = function() {
2     //todo create or update
3     this.trigger('change');
4 };
5 Model.prototype.delete = function() {
6     //todo delete
7     this.trigger('destroy');
8 }

在页面中,我们会根据得到的数据来生成对应的模型,因为我们的模型已经扩展了发布订阅模式,因此在代码中,我们给我们的模型添加订阅事件的代码:

1 //子视图
2 function View(data) {
3     var model = new Model(data);
4     model.bind('change', this.render);
5 }
6 View.prototype.render = function() {
7     //todo view render
8 };

上面便是前端ORM的一个基础框架模型,也是核心点所在,不过要完成整个框架,仍然需要添加很多的东西。
上面的代码仅仅是能实现唯一的模型实例,而不能对模型实例集合进行操作,什么意思呢,比如你的数据是一个数组,
里面包含了多条数据,没一条数据都能转换为一个对象的模型对象。在现在的代码中,我们只完成了一条数据的转换而没有完成整个数据数据的转换,因此,需要做出一些调整;
在服务器端获取回来的数据,我们应该将之转换为一个模型,数据集里面的每一个数据记录,就用来转换成为该模型的每一个实例,按照这个思路,我们的代码使用起来可能像下面这个样子:

1 var User = Model.create();
2 var user = User.init();

模型的创建,也就是User的创建,我们是不需要传递数据的,只需要传递我们需要的一些数据字段即可,这些字段也就是模型实例的一些内置属性;根据User来生成的实例都会具有User所具有的一些信息,在这里就是字段属性。
同时Model还应该具有一些静态方法用来查找实例,修改实例等,因此该Model还要有一个用于保存模型实例的容器,也就是一个数组或者对象。

1 Model.extend({
 2     records: {}, //保存实例
 3     find: function(id) {
 4 
 5     },
 6     exists: function(id) {
 7 
 8     },
 9     ...
10 });

至此,一个大致的ORM构建思路就算是完成了,具体的代码还需要不断的添加完成。