“写的不是代码,是你的思维”,经常觉得自己写的代码“皮粗肉糙”的,看到那些要么精致小巧,要么优雅大方,要么光明磊落的代码时,常常会黯然神伤外加垂涎欲滴。

  why?(为什么我的代码不能如此。。)

  when?(什么时候我的代码能有此质量)

  what?(我要怎么做?)

  此时我的回答是:静下心来,多思考,多实践。(老李, 。。。。。说好的魂淡要做到)

  今天想重新思考实践一下MVC

开场啦。。。。。。

//模型

//封装与应用程序的业务逻辑相关的数据以及数据的处理方法。模型不依赖“视图”和“控制器”//当model发生变化是时,通知它的监听者

function ListModel(items){

  this._items = items;

  this._selectedIndex = -1;

  this.itemAdded = new Event(this);

  this.itemRemoved = new Event(this);

  this.selectedIndexChanged = new Event(this);

}

ListModel.prototype = {

  getItems: function(){

    return [].concat(this._items);

  },

  addItem: function(item){

    this._items.push(item);

    this.itemAdded.notify({item:item});

  },

  removeItemAt: function(index){

    var item;

    item = this._items[index];

    this._items.splice(index, 1);

    this.itemRemoved.notify({item:item});

    if(index === this._selectedIndex){

      this.setSelectedIndex(-1);

    }

  },

  getSelectedIndex: function(){

    return this._selectedIndex;

  },

  setSelectedIndex: function(index){
            var previousIndex;
            previousIndex = this._selectedIndex;
            this._selectedIndex = index;
            this.selectedIndexChanged.notify({previous: previousIndex});
     }

};

function Event(sender){
            this._sender = sender;
            this._listeners = [];
        }
        Event.prototype = {
            attach: function(listener){
                this._listeners.push(listener);
            },
            notify: function(args){
                var index;
                for(index=0;index<this._listeners.length;index +=1){
                    this._listeners[index](this._sender, args);
                }
            }
        };

  //视图
        //视图显示模型数据,并触发模型数据。控制器用来处理这些用户交互事件
        function ListView(model, elements){
            this._model= model;
            this._elements = elements;

this.listModified = new Event(this);
            this.addButtonClicked = new Event(this);
            this.delButtonClicked = new Event(this);

var _this = this;
            //绑定模型监听器
            this._model.itemAdded.attach(function(){
                _this.rebuildList();
            });
            this._model.itemRemoved.attach(function(){
                _this.rebuildList();
            });

//将监听器绑定到HTML控件上
            this._elements.list.change(function(e){
                _this.listModified.notify({index: e.target.selectedIndex});
            });
            this._elements.addButton.click(function () {  
                _this.addButtonClicked.notify();  
            });  
        
            this._elements.delButton.click(function () {  
                _this.delButtonClicked.notify();  
            });

}
        ListView.prototype = {
            show: function(){
                this.rebuildList();
            },
            rebuildList: function(){
                var list, items, key;
                list = this._elements.list;
                list.html("");

items = this._model.getItems();
                for(key in items){
                    if(items.hasOwnProperty(key)){
                        list.append($("<option>"+items[key]+"</option>"));
                    }
                }
                this._model.setSelectedIndex(-1);
            }
        };

//控制器
        //控制器响应用户操作,调用模型上的变化函数
        function ListController(model, view){
            this._model = model;
            this._view = view;

var _this = this;
             this._view.listModified.attach(function (sender, args) {  
                _this.updateSelected(args.index);  
            });  
        
            this._view.addButtonClicked.attach(function () {  
                _this.addItem();  
            });  
        
            this._view.delButtonClicked.attach(function () {  
                _this.delItem();  
            });

}
        ListController.prototype = {
            addItem: function(){
                var item = window.prompt('Add item:','');
                if(item){
                    this._model.addItem(item);
                }
            },
            delItem: function(){
                var index;
                index = this._model.getSelectedIndex();
                if(index!== -1){
                    this._model.removeItemAt(this._model.getSelectedIndex());
                }
            },
            updateSelected: function(index){
                this._model.setSelectedIndex(index);
            }
        };

//}());

  //测试用例
    $(function () {
            var model = new ListModel(['PHP', 'JavaScript']),

view = new ListView(model, {
                'list' : $('#list'),
                'addButton' : $('#plusBtn'),
                'delButton' : $('#minusBtn')
            }),

controller = new ListController(model, view);        
            view.show();
        });

javascript MVC(每天有学习一点篇)的更多相关文章

  1. javascript解析引擎(每天有学习一点篇)

    ======================================================= 有一段时间,经常耳闻web前端的福音,对高性能的V8议论纷纷. 其实对js解析引擎没有深 ...

  2. DOM编程(每天有学习一点篇)

    每次想到“DOM”编程就会自然联想到“性能瓶颈”.我觉得有两部分原因: 1.DOM自己本身操作代价昂贵,因为浏览器通常要求DOM 实现和JavaScript 实现保持相互独立: 2.嘿嘿,本人自身的原 ...

  3. ASP.NET MVC Web API 学习笔记---第一个Web API程序

    http://www.cnblogs.com/qingyuan/archive/2012/10/12/2720824.html GetListAll /api/Contact GetListBySex ...

  4. JavaScript MVC框架PK:Angular、Backbone、CanJS与Ember

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  5. JavaScript MVC框架PK:Angular、Backbone、CanJS与Ember(转载)

    原文地址:http://sporto.github.io/.../comparison-angular-backbone-can-ember/ 原文作者:Sebastian Porto @Twitte ...

  6. MVC系列——MVC源码学习:打造自己的MVC框架(二:附源码)

    前言:上篇介绍了下 MVC5 的核心原理,整篇文章比较偏理论,所以相对比较枯燥.今天就来根据上篇的理论一步一步进行实践,通过自己写的一个简易MVC框架逐步理解,相信通过这一篇的实践,你会对MVC有一个 ...

  7. MVC系列——MVC源码学习:打造自己的MVC框架(一:核心原理)

    前言:最近一段时间在学习MVC源码,说实话,研读源码真是一个痛苦的过程,好多晦涩的语法搞得人晕晕乎乎.这两天算是理解了一小部分,这里先记录下来,也给需要的园友一个参考,奈何博主技术有限,如有理解不妥之 ...

  8. MVC缓存OutPutCache学习笔记 (二) 缓存及时化VaryByCustom

    <MVC缓存OutPutCache学习笔记 (一) 参数配置> 本篇来介绍如何使用 VaryByCustom参数来实现缓存的及时化.. 根据数据改变来及时使客户端缓存过期并更新.. 首先更 ...

  9. 【JavsScript】JavaScript MVC 框架技术选型

    你很喜欢Gmail和Trello之类的单页面应用,但是不太确定该从何开始.也许你的JavaScript代码是如此的杂乱无章,以致于你很想在下一个项目上尝试下JavaScript MVC库和框架,却苦于 ...

随机推荐

  1. php中 -> 和 => 和 :: 的用法 以及 self 和 $this 的用法

    => 数组中 用于数组的 key 和 value之间的关系例如:$a = array( '0' => '1', '2' => '4',); echo $a['0'];echo $a[ ...

  2. 利用ClouderaManager启动HBase时,出现 master.TableNamespaceManager: Namespace table not found. Creating...

    1.错误描述: 出现上述这个错误的原因是我之前已经安装了Cloudera Manager中的CDH,其中添加了所有的服务,当然也包含HBase.然后重新安装的时候,就会出现如下错误: Failed t ...

  3. disconnected no supported authentication methods available(server sent: publickey,keyboard interae)

    因为乌龟Git和Git的冲突 我们需要把乌龟Git设置改正如下. 找到TortoiseGit -> Settings -> Network 将SSH client指向~\Git\bin\s ...

  4. C/C++语言 预处理小结

    预处理功能主要包括宏定义,文件包含,条件编译三部分.分别对应宏定义命令,文件包含命令,条件编译命令三部分实现. 预处理过程读入源代码,检查包含预处理指令的语句和宏定义,并对源代码进行响应的转换.预处理 ...

  5. windows核心编程---第六章 线程的调度

    每个线程都有一个CONTEXT结构,保存在线程内核对象中.大约每隔20ms windows就会查看所有当前存在的线程内核对象.并在可调度的线程内核对象中选择一个,将其保存在CONTEXT结构的值载入c ...

  6. js中退出语句break,continue和return 比较 (转载)

    在 break,continue和return 三个关键字中, break,continue是一起的,return 是函数返回语句,但是返回的同时也将函数停止 首先:break和continue两个一 ...

  7. [转]一个简单的Linux多线程例子 带你洞悉互斥量 信号量 条件变量编程

    一个简单的Linux多线程例子 带你洞悉互斥量 信号量 条件变量编程 希望此文能给初学多线程编程的朋友带来帮助,也希望牛人多多指出错误. 另外感谢以下链接的作者给予,给我的学习带来了很大帮助 http ...

  8. ng-repeat指令中使用track by子语句解决重复数据遍历的错误

    用ng-repeat指令遍历一个javascript数组,当数组中有重复元素的时候,angularjs会报错,这是因为ng-Repeat不允许collection中存在两个相同Id的对象. 对于数字或 ...

  9. OC 属性

    1 属性和实例变量 :属性 ==  实例变量声明 + setter 方法+ getter 方法 在老版本的 OC 语言中,我们需要同时声明属性和底层实例变量,那时,属性是 OC 语言的一个新的机制,并 ...

  10. 源码分析:Java对象的内存分配

    Java对象的分配,根据其过程,将其分为快速分配和慢速分配两种形式,其中快速分配使用无锁的指针碰撞技术在新生代的Eden区上进行分配,而慢速分配根据堆的实现方式.GC的实现方式.代的实现方式不同而具有 ...