亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

JavaScript的Module模式編程深入分析

2019-11-20 22:25:07
字體:
來源:轉載
供稿:網友

基礎知識

首先我們要大概了解一下Module模式(2007年由YUI的EricMiraglia在博客中提出),如果你已熟悉 Module 模式,可以跳過本部分,直接閱讀"高級模式"。

匿名函數閉包

匿名函數閉包是JavaScript最棒的特征,沒有之一,是它讓一切都成為了可能?,F在我們來創建一個匿名函數然后立即執行。函數中所有的代碼都是在一個閉包中執行的,閉包決定了在整個執行過程中這些代碼的私有性和狀態。

復制代碼 代碼如下:

(function () {
 // ... all vars and functions are in this scope only
 // still maintains access to all globals
}());

注意在匿名函數外面的括號。這是由于在JavaScript中以function開頭的語句通常被認為是函數聲明。加上了外面的括號之后則創建的是函數表達式。

全局導入

JavaScript有一個特征叫做隱藏的全局變量。當一個變量名被使用,編譯器會向上級查詢用var來聲明這個變量的語句。如果沒有找到的話這個變量就被認為是全局的。如果在賦值的時候這樣使用,就會創建一個全局的作用域。這意味著在一個匿名的閉包中創建一個全局變量是十分容易的。不幸的是 ,這將會導致代碼的難以管理,因為對于程序員來說,如果全局的變量不是在一個文件中聲明會很不清晰。幸運的是 ,匿名函數給我我們另一個選擇。我們可以將全局變量通過匿名函數的參數來導入到我們的代碼中,這樣更加的快速和整潔。

復制代碼 代碼如下:

(function ($, YAHOO) {
// now have access to globals jQuery (as $) and YAHOO in this code
}(jQuery, YAHOO));

Module導出

有時你并不想要使用全局變量,但是你想要聲明他們。我們可以很容易通過匿名函數的返回值來導出他們。關于Module模式的基本內容就這么多,這里有一個復雜一點的例子。

復制代碼 代碼如下:

var MODULE = (function () {
 var my = {},
  privateVariable = 1;

 function privateMethod() {
  // ...
 }

 my.moduleProperty = 1;
 my.moduleMethod = function () {
  // ...
 };

 return my;
}());

這里我們聲明了一個全局的module叫做MODULE,有兩個公有屬性:一個叫做MODULE.moduleMethod的方法和一個叫做MODULE.moduleProperty的變量。另外他通過匿名函數的閉包來維持私有的內部狀態,當然我們也可使用前面提到的模式,輕松導入所需的全局變量

高級模式

之前提到的內容已經可以滿足很多需求了,但我們可以更深入地研究這種模式來創造一些強力的可拓展的結構。讓我們一點一點,繼續通過這個叫做MODULE的module來學習。

拓展

目前,module模式的一個局限性就是整個module必須是寫在一個文件里面的。每個進行過大規模代碼開發的人都知道將一個文件分離成多個文件的重要性。幸運的是我們有一個很好的方式來拓展modules。首先我們導入一個module,然后加屬性,最后將它導出。這里的這個例子,就是用上面所說的方法來拓展MODULE。

復制代碼 代碼如下:

var MODULE = (function (my) {
 my.anotherMethod = function () {
  // added method...
 };

 return my;
}(MODULE));

雖然不必要,但是為了一致性 ,我們再次使用var關鍵字。然后代碼執行,module會增加一個叫做MODULE.anotherMethod的公有方法。這個拓展文件同樣也維持著它私有的內部狀態和導入。

松拓展

我們上面的那個例子需要我們先創建module,然后在對module進行拓展,這并不是必須的。異步加載腳本是提升 Javascript 應用性能的最佳方式之一。。通過松拓展,我們創建靈活的,可以以任意順序加載的,分成多個文件的module。每個文件的結構大致如下

復制代碼 代碼如下:

var MODULE = (function (my) {
 // add capabilities...

 return my;
}(MODULE || {}));

在這種模式下,var語句是必須。如果導入的module并不存在就會被創建。這意味著你可以用類似于LABjs的工具來并行加載這些module的文件。

緊拓展

雖然松拓展已經很棒了,但是它也給你的module增添了一些局限。最重要的一點是,你沒有辦法安全的重寫module的屬性,在初始化的時候你也不能使用其他文件中的module屬性(但是你可以在初始化之后運行中使用)。緊拓展包含了一定的載入順序,但是支持重寫,下面是一個例子(拓展了我們最初的MODULE)。

復制代碼 代碼如下:

var MODULE = (function (my) {
 var old_moduleMethod = my.moduleMethod;

 my.moduleMethod = function () {
  // method override, has access to old through old_moduleMethod...
 };

 return my;
}(MODULE));

這里我們已經重寫了MODULE.moduleMethod,還按照需求保留了對原始方法的引用。

復制和繼承

復制代碼 代碼如下:

var MODULE_TWO = (function (old) {
 var my = {},
  key;

 for (key in old) {
  if (old.hasOwnProperty(key)) {
   my[key] = old[key];
  }
 }

 var super_moduleMethod = old.moduleMethod;
 my.moduleMethod = function () {
  // override method on the clone, access to super through super_moduleMethod
 };

 return my;
}(MODULE));

這種模式可能是最不靈活的選擇。雖然它支持了一些優雅的合并,但是代價是犧牲了靈巧性。在我們寫的代碼中,那些類型是對象或者函數的屬性不會被復制,只會以一個對象的兩份引用的形式存在。一個改變,另外一個也改變。對于對象來說[g5] ,我們可以通過一個遞歸的克隆操作來解決,但是對于函數是沒有辦法的,除了eval。然而,為了完整性我還是包含了它。

跨文件的私有狀態

把一個module分成多個文件有一很大的局限,就是每一個文件都在維持自身的私有狀態,而且沒有辦法來獲得其他文件的私有狀態。這個是可以解決的,下面這個松拓展的例子,可以在不同文件中維持私有狀態。

復制代碼 代碼如下:

var MODULE = (function (my) {
 var _private = my._private = my._private || {},
  _seal = my._seal = my._seal || function () {
   delete my._private;
   delete my._seal;
   delete my._unseal;
  },
  _unseal = my._unseal = my._unseal || function () {
   my._private = _private;
   my._seal = _seal;
   my._unseal = _unseal;
  };

 // permanent access to _private, _seal, and _unseal

 return my;
}(MODULE || {}));

每一個文件可以為它的私有變量_private設置屬性,其他文件可以立即調用。當module加載完畢,程序會調用MODULE._seal(),讓外部沒有辦法接觸到內部的 _.private。如果之后module要再次拓展,某一個屬性要改變。在載入新文件前,每一個文件都可以調用_.unsea(),,在代碼執行之后再調用_.seal。

這個模式在我今天的工作中想到的,我從沒有在其他地方見到過。但是我認為這是一個很有用的模式,值得單獨寫出來。

Sub-modules

最后一個高級模式實際上是最簡單的,有很多創建子module的例子,就像創建一般的module一樣的。

復制代碼 代碼如下:

MODULE.sub = (function () {
 var my = {};
 // ...

 return my;
}());

雖然這可能是很簡單的,但是我決定這值得被寫進來。子module有一般的module所有優質的特性,包括拓展和私有狀態。

總結

大多數高級模式都可以互相組合來創建更有用的新模式。如果一定要讓我提出一個設計復雜應用的方法的話,我會結合松拓展,私有狀態,和子module。

在這里我沒有提到性能相關的事情,但是我可以說,module模式對于性能的提升有好處。它可以減少代碼量,這就使得代碼的載入更迅速。松拓展使得并行加載成為可能,這同樣提升的載入速度。初始化的時間可能比其他的方法時間長,但是這多花的時間是值得的。只要全局變量被正確導入了運行的時候就不會出問題,在子module中由于對變量的引用鏈變短了可能也會提升速度。

最后,這是一個子module自身動態加載的例子(如果不存在就創建),為了簡介我沒有考慮內部狀態,但是即便考慮它也很簡單。這個模式可以讓復雜,多層次的代碼并行的加載,包括子module和其他所有的東西。

復制代碼 代碼如下:

var UTIL = (function (parent, $) {
 var my = parent.ajax = parent.ajax || {};

 my.get = function (url, params, callback) {
  // ok, so I'm cheating a bit
  return $.getJSON(url, params, callback);
 };

 // etc...

 return parent;
}(UTIL || {}, jQuery));

我希望這些內容是有用的,請在下面留言來分享你的想法。少年們,努力吧,寫出更好的,更模塊化的JavaScript。

原文鏈接:ben cherry 翻譯:王筱

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久久久久久影院| 日韩经典第一页| 欧美精品免费看| 亚洲国产精品人久久电影| 欧美视频国产精品| 国产精品第3页| 日韩成人网免费视频| 91久久中文字幕| 成人免费观看a| 亚洲国产精品一区二区三区| 国产精品色婷婷视频| 亚洲激情第一页| 国产一区二区三区欧美| 亚洲免费成人av电影| 欧日韩在线观看| 日韩中文字幕不卡视频| 国产一区香蕉久久| 国产999精品久久久影片官网| 中文字幕九色91在线| 亚洲女人天堂成人av在线| 欧美激情xxxx性bbbb| 粉嫩av一区二区三区免费野| 亚洲国产91精品在线观看| 在线一区二区日韩| 精品中文字幕在线| 欧美日韩中文字幕综合视频| 亚洲精品网址在线观看| 欧美黑人巨大xxx极品| 欧美精品成人在线| 久久久人成影片一区二区三区观看| 成人网欧美在线视频| 91在线网站视频| 九九精品视频在线| 国产精品国内视频| 国产91对白在线播放| 欧美成人全部免费| 国模精品视频一区二区| 久久久久久久久久久免费| 影音先锋欧美在线资源| 欧美大片免费看| 91老司机在线| 国产精品一区二区久久国产| 欧美xxxwww| 午夜精品一区二区三区在线视频| 亚洲成人在线视频播放| 尤物99国产成人精品视频| 欧美精品久久久久久久久| 这里只有精品久久| 91极品视频在线| 亚洲国产美女精品久久久久∴| 国产一区二区三区精品久久久| 亚洲精品电影久久久| 久久91亚洲精品中文字幕| 日韩精品极品在线观看| 久久成人这里只有精品| 日本久久久久久久久久久| 精品久久久久久久久久久久久| 日韩av网站大全| 欧美—级a级欧美特级ar全黄| 欧美亚洲国产精品| 亚洲人成伊人成综合网久久久| 欧美韩国理论所午夜片917电影| 亚洲成人性视频| 欧美国产日韩一区二区三区| 91精品久久久久久久久青青| 亚洲天堂av综合网| 亚洲欧美精品中文字幕在线| 日韩在线中文视频| 77777少妇光屁股久久一区| 久久精品亚洲一区| 一级做a爰片久久毛片美女图片| 亚洲sss综合天堂久久| 欧美午夜精品久久久久久浪潮| 欧洲成人在线视频| 欧美一级高清免费| 日韩中文字幕在线免费观看| 高潮白浆女日韩av免费看| 久青草国产97香蕉在线视频| 国产日韩欧美中文| 亚洲精品美女久久久| 在线精品视频视频中文字幕| 91在线观看免费高清完整版在线观看| 国产在线精品自拍| 国产精品高潮呻吟久久av黑人| 国产日韩在线免费| 亚洲一区二区自拍| 国产精品永久免费视频| 亚洲在线一区二区| 萌白酱国产一区二区| 国模精品视频一区二区三区| 亚洲视频在线免费看| 欧美日韩国产中字| 色偷偷偷综合中文字幕;dd| 久久午夜a级毛片| 日本精品视频在线| 亚洲成人教育av| 欧美日韩免费观看中文| 川上优av一区二区线观看| 成人中文字幕+乱码+中文字幕| 久久精品亚洲精品| 国产精品99蜜臀久久不卡二区| 国产中文字幕91| 国产黑人绿帽在线第一区| 亚洲欧美成人一区二区在线电影| 日韩大陆欧美高清视频区| 91亚洲精品在线| 亚洲欧洲中文天堂| 国产一区二区三区高清在线观看| 日本精品视频网站| 免费91在线视频| 亚洲福利视频专区| 夜夜嗨av一区二区三区免费区| 欧美成人午夜激情视频| 精品国产1区2区| 国产精品亚洲自拍| 九九热精品视频| 亚洲第一天堂无码专区| 精品国产一区二区三区四区在线观看| 欧美有码在线观看视频| 亚洲精品国产精品乱码不99按摩| 日韩乱码在线视频| 黄网站色欧美视频| 欧美电影在线观看网站| 91干在线观看| 国产精品美女av| 国产精品偷伦一区二区| 91在线免费视频| 欧美美女18p| 国产剧情日韩欧美| 日本一区二区不卡| 亚洲石原莉奈一区二区在线观看| 日韩视频亚洲视频| 日韩欧美国产骚| 国产欧美精品一区二区三区介绍| 欧美影院在线播放| 欧美日本黄视频| 成人午夜在线影院| 在线日韩第一页| 欧美性生交xxxxx久久久| 亚洲欧美国产va在线影院| 久久久久久久影视| 欧美寡妇偷汉性猛交| 国产日韩av高清| 久久久久久免费精品| 国产91精品久| xxx成人少妇69| 欧美成人久久久| www.久久色.com| 国产亚洲精品成人av久久ww| 久久99视频精品| 国产精品三级网站| 亚洲精品98久久久久久中文字幕| 亚洲成年网站在线观看| 亚洲成年人影院在线| 琪琪第一精品导航| 欧美日韩在线观看视频小说| 热99在线视频| 亚洲精品电影久久久| 97精品视频在线观看| 色偷偷偷综合中文字幕;dd| www.xxxx欧美| 欧美中文字幕在线视频| 91精品久久久久| www.xxxx欧美|