JavaScript插件化开发
大熊君JavaScript插件化开发
一,开篇分析
Hi,大家好!大熊君又和大家见面了,还记得昨天的那篇文章吗------这个系列的开篇(第一季)。主要讲述了以“jQuery的方式如何开发插件”,
那么今天我们带着昨天的疑问来继续我们的插件开发之旅。昨天的问题如下:
(1),如果项目技术选型换了这些插件又是强依赖“jQuery”机制,我们以前写的插件将会不能用(假设不用jQuery的情况),如何做重构那?
(2),重构插件的关键逻辑,我们将如何组织那?
好了,带着问题去学习今天的文章吧。
首先我不是否定“jQuery插件的方式”,其次是我们要从不同的角度分析问题,比如说“jQuery插件有如下优点”:
(1),把全部代码放在闭包(一个即时执行函数)里此时闭包相当于一个私有作用域,外部无法访问到内部的信息,并且不会存在全局变量的污染情况。
(2),a) 避免全局依赖;b) 避免第三方破坏;c) 兼容jQuery操作符'$'和'jQuery '。
那我们重构将以什么方式组织代码那,是面向对象的思想(OOP)那?还是过程化的思路进行到底那?还是两者结合设计那?哈哈哈,继续看。。。。。。
二,重构昨天的例子
以下是昨天的Js部分源码部分:

1 (function($){
2 $.fn.bigbear = function(opts){
3 opts = $.extend({},$.fn.bigbear.defaults,opts) ;
4 return this.each(function(){
5 var elem = $(this) ;
6 elem.find("span").text(opts["title"]) ;
7 $.get(opts["url"],function(data){
8 elem.find("div").text(data["text"]) ;
9 }) ;
10 }) ;
11 } ;
12 $.fn.bigbear.defaults = {
13 title : "这是一个简单的测试" ,
14 url : "data.json"
15 } ;
16 })(jQuery) ;

我们来逐行分析一下:
首先确定一下这个插件的功能
(1),显示我们设置的标题文字信息。
(2),动态通过异步的方式获取内容信息。
好了!需求明确就好展开讨论了,从上面的代码不难看出逻辑组织很松散,过程化的思维很明显,所以第一步就是把我们的功能需求
以类的方式有效地组织起来。看如下重构后的代码:

1 $(function(){
2 $("#bb").bigbear() ;
3 }) ;
4 (function($){
5 $.fn.bigbear = function(opts){
6 opts = $.extend({},$.fn.bigbear.defaults,opts) ;
7 return this.each(function(){
8 var elem = $(this) ;
9 var bb = new BigBear(elem,opts) ;
10 bb.getElem().trigger("data") ;
11 }) ;
12 } ;
13 $.fn.bigbear.defaults = {
14 title : "这是一个简单的测试" ,
15 url : "data.json"
16 } ;
17 })(jQuery) ;
18
19 function BigBear(elem,opts){
20 this.elem = elem ;
21 this.opts = opts ;
22 this.init() ;
23 } ;
24 var bbProto = BigBear.prototype ;
25 bbProto.getElem = function(){
26 return this.elem ;
27 } ;
28 bbProto.getOpts = function(){
29 return this.opts ;
30 } ;
31 bbProto.init = function(){
32 var that = this ;
33 this.getElem().on("data",function(){
34 that._setTitle(that.getOpts()["title"]) ;
35 $.get(that.getOpts()["url"],function(result){
36 that.getElem().find("div").text(result["text"]) ;
37 }) ;
38 }) ;
39 } ;
40 bbProto._setTitle = function(text){
41 this.getElem().find("span").text(text) ;
42 } ;

哈哈哈,是不是代码多了不少,其实这种方式就是面向对象的角度看问题,先去分析功能需求,然后设计我们的类,虽然说我们不可能一下设计得很出色,
但是看问题角度改变了,我们的代码可读性强了,以及更好地进行维护这样我们的目的也就达到了。
以下是是摘自“Bootstrap”Js部分的相关源码实现,如下图:

不难看出也是相似的实现方式,通过类来维护我们插件的主要逻辑。
(三),增加新功能,引出额外的类
现在需求增加了,需要在体验上有所变化,加载数据时有“loading”效果。
实现思路可以这样,在原始的内容区把文字设置成“装载数据中。。。。”的字样,然后引入一个新的类,如下:

1 function Overlay(){
2
3 } ;
4 var olProto = Overlay.prototype ;
5 olProto.show = function(){} ;
6 olProto.hide = function(){} ;
7 // 具体实现就不写了

好了,遮罩层已经有了,现在我们怎么集成进来那?我们用组合的方式接入进来,如下:

1 function BigBear(elem,opts){
2 this.elem = elem ;
3 this.opts = opts ;
4 this.overlay = new Overlay() ;
5 this.init() ;
6 } ;
7 var bbProto = BigBear.prototype ;
8 bbProto.getElem = function(){
9 return this.elem ;
10 } ;
11 bbProto.getOpts = function(){
12 return this.opts ;
13 } ;
14 bbProto.init = function(){
15 var that = this ;
16 var loadingText = "数据装载中。。。" ;
17 this.getElem().on("data",function(){
18 that._setTitle(that.getOpts()["title"]) ;
19 that.overlay.show() ;
20 that.getElem().find("div").text(loadingText) ;
21 $.get(that.getOpts()["url"],function(result){
22 that.overlay.hide() ;
23 that.getElem().find("div").text(result["text"]) ;
24 }) ;
25 }) ;
26 } ;
27 bbProto._setTitle = function(text){
28 this.getElem().find("span").text(text) ;
29 } ;

到此只为我们的功能就算是结束了,这样写的插件,我相信比第一个版本好很多,当然这不是最优的实现,需要从细节上不断重构,但是这种方式是一种可选的开发插件的方式。
以下是完整的代码:

1 $(function(){
2 $("#bb").bigbear() ;
3 }) ;
4 (function($){
5 $.fn.bigbear = function(opts){
6 opts = $.extend({},$.fn.bigbear.defaults,opts) ;
7 return this.each(function(){
8 var elem = $(this) ;
9 var bb = new BigBear(elem,opts) ;
10 bb.getElem().trigger("data") ;
11 }) ;
12 } ;
13 $.fn.bigbear.defaults = {
14 title : "这是一个简单的测试" ,
15 url : "data.json"
16 } ;
17 })(jQuery) ;
18
19 function BigBear(elem,opts){
20 this.elem = elem ;
21 this.opts = opts ;
22 this.overlay = new Overlay() ;
23 this.init() ;
24 } ;
25 var bbProto = BigBear.prototype ;
26 bbProto.getElem = function(){
27 return this.elem ;
28 } ;
29 bbProto.getOpts = function(){
30 return this.opts ;
31 } ;
32 bbProto.init = function(){
33 var that = this ;
34 var loadingText = "数据装载中。。。" ;
35 this.getElem().on("data",function(){
36 that._setTitle(that.getOpts()["title"]) ;
37 that.overlay.show() ;
38 that.getElem().find("div").text(loadingText) ;
39 $.get(that.getOpts()["url"],function(result){
40 that.overlay.hide() ;
41 that.getElem().find("div").text(result["text"]) ;
42 }) ;
43 }) ;
44 } ;
45 bbProto._setTitle = function(text){
46 this.getElem().find("span").text(text) ;
47 } ;
48
49 function Overlay(){
50
51 } ;
52 var olProto = Overlay.prototype ;
53 olProto.show = function(){} ;
54 olProto.hide = function(){} ;
55 // 具体实现就不写了

(四),最后总结
(1),面向对象的思考方式合理分析功能需求。
(2),以类的方式来组织我们的插件逻辑。
(3),不断重构上面的实例,最好自己写一下,并且完善“Overlay”的逻辑。
(4),大家学习完继续思考一下,如何进行合理的重构那?不要设计过度,要游刃有余,推荐的方式是过程化设计与面向对象思想设计相结合。
哈哈哈,本篇结束,未完待续,希望和大家多多交流够沟通,共同进步。。。。。。呼呼呼……(*^__^*)
JavaScript插件化开发的更多相关文章
- 大熊君JavaScript插件化开发------(第一季)
一,开篇分析 Hi,大家!大熊君又来了,今天这系列文章主要是说说如何开发基于“JavaScript”的插件式开发,我想很多人对”插件“这个词并不陌生, 有的人可能叫“组件”或“部件”,这不重要,关键是 ...
- 大熊君JavaScript插件化开发------(实战篇之DXJ UI ------ ItemSelector重构完结版)
一,开篇分析 Hi,大家好!大熊君又和大家见面了,还记得上一篇文章吗.主要讲述了以“jQuery的方式如何开发插件”,以及过程化设计与面向对象思想设计相结合的方式是 如何设计一个插件的,两种方式各有利 ...
- 大熊君JavaScript插件化开发------(实战篇之DXJ UI ------ ItemSelector)
一,开篇分析 Hi,大家好!大熊君又和大家见面了,还记得前两篇文章吗.主要讲述了以“jQuery的方式如何开发插件”,以及过程化设计与面向对象思想设计相结合的方式是 如何设计一个插件的,两种方式各有利 ...
- 大熊君JavaScript插件化开发------(实战篇之DXJ UI ------ ProcessBar)
一,开篇分析 Hi,大家好!大熊君又和大家见面了,还记得前两篇文章吗.主要讲述了以“jQuery的方式如何开发插件”,以及过程化设计与面向对象思想设计相结合的方式是 如何设计一个插件的,两种方式各有利 ...
- 大熊君JavaScript插件化开发------(实战篇之DXJ UI ------ Tab)
一,开篇分析 Hi,大家好!大熊君又和大家见面了,还记得前两篇文章吗.主要讲述了以“jQuery的方式如何开发插件”,以及过程化设计与面向对象思想设计相结合的方式是 如何设计一个插件的,两种方式各有利 ...
- 大熊君JavaScript插件化开发------(第二季)
一,开篇分析 Hi,大家好!大熊君又和大家见面了,还记得昨天的那篇文章吗------这个系列的开篇(第一季).主要讲述了以“jQuery的方式如何开发插件”, 那么今天我们带着昨天的疑问来继续我们的插 ...
- 大熊君JavaScript插件化开发------(实战篇之DXJ UI ------ Tab功能扩展完结版)
一,开篇分析 Hi,大家好!大熊君又和大家见面了,还记得上一篇文章吗.主要讲述了一个“Tab”插件是如何组织代码以及实现的”,以及过程化设计与面向对象思想设计相结合的方式是 如何设计一个插件的,两种方 ...
- TinyFrame升级之八:实现简易插件化开发
本章主要讲解如何为框架新增插件化开发功能. 在.net 4.0中,我们可以在Application开始之前,通过PreApplicationStartMethod方法加载所需要的任何东西.那么今天我们 ...
- Android组件化和插件化开发
http://www.cnblogs.com/android-blogs/p/5703355.html 什么是组件化和插件化? 组件化开发就是将一个app分成多个模块,每个模块都是一个组件(Modul ...
随机推荐
- 版本管理软件VisualSVN、TortoiseSvn、AnkhSvn 后记
原文:版本管理软件VisualSVN.TortoiseSvn.AnkhSvn 后记 前些天我写了几篇关于VisualSVN .TortoiseSVN.AnkhSvn这几个软件配置管理的文章,但是当时没 ...
- Java实现 Base64、MD5、MAC、HMAC加密(转)
开始对那些基本的加密还不怎么熟练,然后总结了些,写了一个测试:支持 Base64.MD5.MAC.HMAC加密,长话短说,我们都比较喜欢自己理解,看代码吧! 采用的输UTF-8的格式... packa ...
- AsyncTask測试多任务
本人进行过模拟測试,发现AsyncTask并不适合多任务,以及长期的异步任务,由于每次仅仅能执行一个AsyncTask,假设执行多个其他任务将会等待 以下通过一个代码样例和日志打印得到证实. 以下扩展 ...
- ffmpeg和opencv 播放视频文件和显示器
ffmpeg它是基于最新版本,在官网下载http://ffmpeg.zeranoe.com/builds/.编译时VS2010配置相关头文件及库的路径就可以.opencv的搭建參考上一个博客. 首先简 ...
- 转让malloc()该功能后,发生了什么事内核?附malloc()和free()实现源
特此声明:在本文中,引用另一篇文章和帖子,结合的概括的理解malloc()函数的实现机制. 我们常常会在C程序中调用malloc()函数动态分配一块连续的内存空间并使用它们.那么,这些用户空间发生的事 ...
- JavaScript事件收集
1. onabort . 2. onactivate 当对象设置为活动元素时触发. 3. onafterprint 对象所关联的文档打印或打印预览后马上在对象上触发. 4. onafterupda ...
- VB6基本数据库应用(五):数据的查找与筛选
同系列的第五篇,上一篇在:http://blog.csdn.net/jiluoxingren/article/details/9633139 数据的查找与筛选 第4篇发布到现在已经过了4天,很抱歉,学 ...
- NSIS:获取硬盘中容量最大的分区盘符
原文 NSIS:获取硬盘中容量最大的分区盘符 我们在安装一些在线视频软件比如迅雷看看时,会发现他们的安装程序会自动判断当前系统中容量最大的分区,以便在其中创建数据缓冲下载的文件夹,这种功能如果实现呢, ...
- i++与++i哪个效率更高
简单的比较前缀自增运算符和后缀自增运算符的效率是片面的, 因为存在很多因素影响这个问题的答案. 首先考虑内建数据类型的情况: 如果自增运算表达式的结果没有被使用, 而是仅仅简单地用于增加一元操作数, ...
- Android在第三方应用程序系统应用尽早开始,杀死自己主动的第三方应用程序,以重新启动
1.为什么第三方应用程序可能早于System的app启动? Android能够查阅了,这里就不细述了,这里不阐述ROM启动还有bootloader.软件启动的大致流程应该是 启动kernel 执行se ...