AngularJS入门知识之MVW类框架的编程思想探讨


Posted in Javascript onDecember 08, 2014

本文通过实现两个简单的业务需求,探讨AngularJS和传统的JavaScript控制DOM实现方式的差别,并尝试理解MVW此类框架在流行的Web前端开发中的编程思想。

这个需求很常见,比如,一个两级菜单,在第一级别菜单项点击时候,对应的子菜单项目应该显示或隐藏。

jQuery的实现:

<!-- html -->

<ul class="parent">

    <li class="parent_item">

        Item 1

        <ul class="child">

            <li class="child_item">Item child 1</li>

        </ul>

    </li>

</ul>
// javascript

$('li.parent_item').click(function(){

    $(this).children('ul.child').toggle();

})

AngularJS的实现:

<!-- html -->

<ul>

    <li ng-click="hide_child = !hide_child">

        Item 1

        <ul ng-hide="hide_child">

            <li>Item child 1</li>

        </ul>

    </li>

</ul>

传统操作DOM的方式,不再赘述。AngularJS的实现,相对代码要精炼很多,只有HTML的版本即可。以上代码,用到了AngularJS这些知识点:

1.Directives 

2.Expressions

ng-click和ng-hide都是框架自带的Directives(指令),前者相当于给li标签提供了一个Event Handler,在该HTML元素(li)被点击的时候,会执行hide_child = !hide_child这个Expression(表达式)。我们先看一下ng-hide这个指令,它会根据赋值的表达式结果(布尔值)来控制该HTML元素是否要显示(通过CSS实现)。也就是说,如果hide_child这个变量如果是true,那么ul就会被隐藏,否则结果相反。  

这里hide_child其实是$scope上的一个变量,对它的值的变更,也可以用controller控制器包装一个方法来实现,只不过现在的语句比较简单,直接写在了指令的赋值里面。

通过以上简单的代码分析,我们可以看到AngularJS两个比较明显的特点:

1.通过指令和表达式对DOM的操作进行了封转,只需简单的代码便可省去额外的JavaScript代码
2.指令和表达式的应用,只直接嵌套在HTML中的,这和jQuery推从的Unobtrusive JavaScript的代码风格有些背道而驰

我们先看另外一个需求,再详细解释上面的结论。

需求2:通过点击div,触发选择form中的一个radio button
传统的HTML Form元素,在如今的移动设备上,操作起来并不是十分友好。比如,Radio button单选框,在触摸屏上,需要精确的位置定位,才能控制好这个组件,但是手指定位又很粗糙。常见的做法,是添加一个对应的Label控件,但是文字本身占屏比例也并不理想,而且也不具备明确的信息传达效果。所以,通常会间接操作一个区域比较大的div或者li标签。

jQuery的实现:

<!-- html -->

<ul>

    <li class="selection">

        <input type="radio"

            id="option1" />

        <label for="option1">option 1</label>

    </li>

</ul>
// javascript

$('li.selection').click(function(){

    $(this).children('input[type="radio"]').click();

})

AngularJS的实现:

<!-- html -->

<ul>

    <li ng-repeat="option in options"

        ng-click="model.option = option.value"

        ng-class="{active: model.option == option.value}" >

        <input type="radio"

            ng-model="model.option"

            value="{{option.value}}"

            id="option1" />

        <label for="option1">option 1</label>

    </li>

</ul>

在这个解决方案中,我们同样没有涉及到额外的JavaScript代码,并且多用了几个指令。为了对比参照,我们只关心ng-click和ng-model这两个指令的表达式。

我们先看一下input这个元素的ng-model指令,这里赋值的意思是,我们把模板上的input和$scope.model对象的option属性进行了关联,深入了解数据绑定可以参考Data Binding。这种指定关联,使得模板控件直接和数据Model进行了绑定,并且这种绑定是双向的。意味着,一旦用户修改控件中的值(勾选radio input),对应的Model对象就会重新赋值(model.option);同时,如果Model对象的值发生了变化,模板中的input控件也会对应反映变化。而这点,在上述jQuery的实现中,其实是没有做到的。

所以,这里通过AngularJS的数据绑定,点击li元素间接完成触发input的流程是这样子的:

1.点击li标签,给model.option赋值;
2.修改了Model对象,定位到对应input控件(value的值为model.option那个);
3.激活input控件的checked属性

通过以上两个案例,我们对Web前端的操作有了新的认识。

首先,技术实现上,通过引入新的指令,表达式,数据绑定等概念,我们可以完全新的方式去操作DOM,而不仅仅局限在用户和HTML组件交互操作上的JavaScript代码的实现。这种思想的变化是巨大的。

从本世纪初,动态Web编程的兴起开始,服务器端的编程技术一直在改进。从一开始的CGI/PHP/ASP,由语言和平台产生了.NET vs. Java,开发效率和软件过程促进了MVC框架/ORM/AOP等,性能和大数据带来了NodeJS/NoSQL/Hadoop等,而浏览器前端的技术需求似乎没有那么激进过。一方面,通过服务器端和数据库,大部分B/S模型的业务需求都能满足;再者,浏览器本身存在不同平台的差异性,对脚本语言和渲染技术的标准不兼容,以及运算能力的欠缺和安全性的考虑。

在这种情况下,浏览器端的需求,大部分时候只需要考虑渲染页面和简单的用户交互。HTML/DOM加上JavaSript/CSS就这样成就了前端的主要工作。所以,以前是没有前端工作师,只需要Web设计师的。慢慢对前端的要求多起来,jQuery成为使用程度最高的一个JavaScript操作DOM的封装库。而在这个阶段,jQuery/JavaScript的主要任务,仍然只是作为面向用户浏览器终端呈现和交互的工具。

理解了jQuery的起源,我们不难发现,以前追求的一些规则,譬如Unobtrusive JavaScript,当时局限于实现的手段和方式,为了分离DOM和JavaScript代码逻辑,我们优先选择了维护性更高的方式。前端对JavaScript的需求加大之后,出现了很多MVC/MVP的前端框架,以及AngularJS所谓的MVW(Model-View-Whatever),JavaScript和DOM一刀切的方式发生了变化。原先我们考虑界面显示和用户交互的直接操作,现在我们有了客户端的数据绑定,丰富的指令,依赖注入,等待我们的将是全新的编程模型和思维方式。 

Javascript 相关文章推荐
JS解密入门 最终变量劫持
Jun 25 Javascript
多种JQuery循环滚动文字图片效果代码
Jun 23 Javascript
极易被忽视的javascript面试题七问七答
Feb 15 Javascript
JQuery Mobile实现导航栏和页脚
Mar 09 Javascript
jQuery判断元素是否显示 是否隐藏的简单实现代码
May 19 Javascript
详解handlebars+require基本使用方法
Dec 21 Javascript
浅谈事件冒泡、事件委托、jQuery元素节点操作、滚轮事件与函数节流
Jul 22 jQuery
关于jquery form表单序列化的注意事项详解
Aug 01 jQuery
vue利用axios来完成数据的交互
Mar 23 Javascript
用node开发并发布一个cli工具的方法步骤
Jan 03 Javascript
vue实现的多页面项目如何优化打包的步骤详解
Jul 19 Javascript
JS Canvas接口和动画效果大全
Apr 29 Javascript
Js中使用hasOwnProperty方法检索ajax响应对象的例子
Dec 08 #Javascript
ECMAScript5中的对象存取器属性:getter和setter介绍
Dec 08 #Javascript
JavaScript对象之深度克隆介绍
Dec 08 #Javascript
SeaJS 与 RequireJS 的差异对比
Dec 08 #Javascript
JavaScript中的ArrayBuffer详细介绍
Dec 08 #Javascript
JS实现仿京东淘宝竖排二级导航
Dec 08 #Javascript
js继承call()和apply()方法总结
Dec 08 #Javascript
You might like
使用adodb lite解决问题
2006/12/31 PHP
基于JQuery框架的AJAX实例代码
2009/11/03 Javascript
侧栏跟随滚动的简单实现代码
2013/03/18 Javascript
JavaScript数字和字符串转换示例
2014/03/26 Javascript
教你如何在Node.js中使用jQuery
2016/08/28 Javascript
js仿QQ邮箱收件人选择与搜索功能
2017/02/10 Javascript
浅谈js中function的参数默认值
2017/02/20 Javascript
Bootstrap modal 多弹窗之叠加关闭阴影遮罩问题的解决方法
2017/02/27 Javascript
Webpack打包css后z-index被重新计算的解决方法
2017/06/18 Javascript
详解react-refetch的使用小例子
2019/02/15 Javascript
9102年webpack4搭建vue项目的方法步骤
2019/02/20 Javascript
深入浅出 Vue 系列 -- 数据劫持实现原理
2019/04/23 Javascript
layui复选框限制选择个数的方法
2019/09/18 Javascript
Vue el-autocomplete远程搜索下拉框并实现自动填充功能(推荐)
2019/10/25 Javascript
JavaScript编写开发动态时钟
2020/07/29 Javascript
vue实现路由懒加载的3种方法示例
2020/09/01 Javascript
[03:12]2016完美“圣”典风云人物:单车专访
2016/12/02 DOTA
[00:27]DOTA2荣耀之路2:Patience from zhou!
2018/05/24 DOTA
python实现去除下载电影和电视剧文件名中的多余字符的方法
2014/09/23 Python
Python中处理字符串的相关的len()方法的使用简介
2015/05/19 Python
用python处理图片之打开\显示\保存图像的方法
2018/05/04 Python
Python字符串逆序输出的实例讲解
2019/02/16 Python
Python socket处理client连接过程解析
2020/03/18 Python
windows10在visual studio2019下配置使用openCV4.3.0
2020/07/14 Python
Django Admin后台模型列表页面如何添加自定义操作按钮
2020/11/11 Python
美国知名玩具品牌:Melissa & Doug
2016/08/16 全球购物
应用化学专业本科生求职信
2013/09/29 职场文书
自荐信怎么写好
2013/11/11 职场文书
村党支部书记个人对照材料汇报
2014/10/26 职场文书
给领导的感谢信范文
2015/01/23 职场文书
大学军训决心书
2015/02/05 职场文书
中学生综合素质自我评价
2015/03/06 职场文书
2016年元旦寄语
2015/08/17 职场文书
Ajax 的初步实现(使用vscode+node.js+express框架)
2021/06/18 Javascript
Python 详解通过Scrapy框架实现爬取百度新冠疫情数据流程
2021/11/11 Python
如何避免mysql启动时错误及sock文件作用分析
2022/01/22 MySQL