[js高手之路]設計模式系列課程-設計一個模塊化擴展功能(define)和使用(use)庫

来源:http://www.cnblogs.com/ghostwu/archive/2017/09/05/7482098.html
-Advertisement-
Play Games

模塊化的誕生標志著javascript開發進入工業時代,近幾年隨著es6, require js( sea js ), node js崛起,特別是es6和node js自帶模塊載入功能,給大型程式開髮帶來了極大的便利。這幾個東西沒有出來之前,最原始的開發全部是利用全局函數進行封裝,如: 這種開發方式 ...


模塊化的誕生標志著javascript開發進入工業時代,近幾年隨著es6, require js( sea js ), node js崛起,特別是es6和node js自帶模塊載入功能,給大型程式開髮帶來了極大的便利。這幾個東西沒有出來之前,最原始的開發全部是利用全局函數進行封裝,如:

1 function checkEmail(){}
2 function checkName(){}
3 function checkPwd(){}

這種開發方式,非常容易造成一個問題,全局變數污染(覆蓋)

如:有個頁面需要引入3個js文件( a.js, b.js, c.js ), A程式員在a.js文件中定義了一個函數叫show, B程式員在b.js中也定義了一個函數叫show, 那麼引入同一個頁面之後,後面的函數會把前面的覆蓋,那C程式員引入之後,本來想調用A程式員的show方法,結果被B程式員寫的文件給覆蓋了,這就很恐怖了,如果這是一個公共方法(核心方法),可能造成系統完全掛掉,那怎麼解決呢?早期程式員一般用命名空間,如:

 1 var ghostwu = ghostwu || {};
 2 ghostwu.tools = {
 3     checkEmail: function () {
 4     },
 5     checkName: function () {
 6     },
 7     checkPwd: function () {
 8     }
 9 }
10 var zhangsan = zhangsan || {};
11 zhangsan.tools = {
12     checkEmail: function () {
13     },
14     checkName: function () {
15     },
16     checkPwd: function () {
17     }
18 }
19 ghostwu.tools.checkPwd();
20 zhangsan.tools.checkPwd();

這樣就有可能大大降低重名的可能性,但是還是會有小部分覆蓋的可能性,當然可以在程式中判斷,如果存在這個命名空間,就不讓定義,但是這樣子寫,就沒有封裝性可言,只要引入了js文件,誰都能夠操作這些命名空間,也很危險,對於公共介面( 比如獲取數據,可以暴露給外部使用),對於一些核心,比較重要的,就要封裝起來,不要暴露給外部使用,所以可以通過閉包( 立即表達式)把命名空間中的方法選擇性的暴露給外部

 1 (function ( window, undefined ) {
 2     var ghostwu = ghostwu || {};
 3     ghostwu.tools = {
 4         checkEmail: function () {
 5             console.log('ghostwu給你暴露一個公共的checkEmail方法');
 6         },
 7         checkName: function () {
 8             console.log('checkName');
 9         },
10         checkPwd: function () {
11             console.log('checkPwd');
12         }
13     },
14     public = {
15         checkEmail : ghostwu.tools.checkEmail,
16     }
17     window.G = public;
18 })( window );
19 G.checkEmail();
20 G.checkName(); //報錯, checkName沒有暴露出來

改進之後,封裝性和團隊協作開發又得到了進一步的提高,但是依然存在另一個問題,後來的開發者如果要為這個框架新增一個功能模塊,而且也能選擇性的暴露一些介面怎麼辦呢?

我們可以封裝一個模塊擴展功能和模塊使用功能,以後要做擴展就非常方便了.

 1 (function (window, undefined) {
 2     var G = G || {};
 3     G.version = 'v1.0.0';
 4     G.define = function (path, fn) {
 5         var args = path.split('.'),
 6             parent = old = this;
 7         if (args[0] === 'G') {
 8             args = args.slice(1);
 9         }
10         if (args[0] === 'define' || args[0] === 'module') {
11             return;
12         }
13         for (var i = 0, len = args.length; i < len; i++) {
14             if (typeof parent[args[i]] === 'undefined') {
15                 parent[args[i]] = {};
16             }
17             old = parent;
18             parent = parent[args[i]];
19         }
20         if (fn) old[args[--i]] = fn();
21         return this;
22     };
23     window.G = G;
24 })(window);
25 
26 G.define( "ghostwu.string", function(){
27     return {
28         trimLeft : function( str ){
29             return str.replace( /^\s+/, '' );
30         },
31         trimRight : function(){
32             return str.replace( /\s+$/, '' );
33         },
34         trim : function(){
35             return str.replace( /^\s+|\s+$/g, '' );
36         }
37     }
38 } );
39 
40 var str = '  跟著ghostwu學習設計模式 ';
41 alert( '(' + str + ')' );
42 alert( '(' + G.ghostwu.string.trimLeft( str ) + ')' );
43 alert( '(' + G.ghostwu.string.trimRight( str ) + ')' );
44 alert( '(' + G.ghostwu.string.trim( str ) + ')' );

我們封裝了一個define函數,這個函數用來給G模塊擴展功能,這裡我擴展了一個字元串處理對象,包含了3個方法: trim, trimLeft, trimRight

使用的時候,按照命名空間的方式使用即可,我們也可以加一個使用模塊的方法use

 1 (function (window, undefined) {
 2     var G = G || {};
 3     G.version = 'v1.0.0';
 4     G.define = function (path, fn) {
 5         var args = path.split('.'),
 6             parent = old = this;
 7         if (args[0] === 'G') {
 8             args = args.slice(1);
 9         }
10         if (args[0] === 'define' || args[0] === 'module') {
11             return;
12         }
13         for (var i = 0, len = args.length; i < len; i++) {
14             if (typeof parent[args[i]] === 'undefined') {
15                 parent[args[i]] = {};
16             }
17             old = parent;
18             parent = parent[args[i]];
19         }
20         if (fn) old[args[--i]] = fn();
21         return this;
22     };
23     G.use = function(){
24         var args = Array.prototype.slice.call( arguments ),
25             fn = args.pop(),
26             path = args[0] && args[0] instanceof Array ? args[0] : args,
27             relys = [],
28             mId = '',
29             i = 0,
30             len = path.length,
31             parent, j, jLen;
32             while ( i < len ) {
33                 if( typeof path[i] === 'string' ){
34                     parent = this;
35                     mId = path[i].replace( /^G\./, '' ).split('.');
36                     for( var j = 0, jLen = mId.length; j < jLen; j++ ){
37                         parent = parent[mId[j]] || false;
38                     }
39                     relys.push( parent );
40                 }else {
41                     relys.push( path[i] );
42                 }
43                 i++;
44             }
45             fn.apply( null, relys );
46     };
47     window.G = G;
48 })(window);
49 
50 G.define( "ghostwu.string", function(){
51     return {
52         trimLeft : function( str ){
53             return str.replace( /^\s+/, '' );
54         },
55         trimRight : function(){
56             return str.replace( /\s+$/, '' );
57         },
58         trim : function(){
59             return str.replace( /^\s+|\s+$/g, '' );
60         }
61     }
62 } );
63 
64 var str = '  跟著ghostwu學習設計模式 ';
65 // G.use( ['ghostwu.string'], function( s ){
66     // console.log( '(' + s.trim( str ) + ')' );
67 // } );
68 // G.use( ['ghostwu.string', document], function( s, doc ){
69     // console.log( doc );
70     // console.log( '(' + s.trim( str ) + ')' );
71 // } );
72 G.use( 'ghostwu.string', function( s ){
73     console.log( s );
74     alert( '(' + s.trim( str ) + ')' );
75 } );

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • margin-left: 40%; ...
  • CSS佈局這種東西既複雜也簡單,弄清楚常見佈局的各種實現方法,方能實現更加複雜的佈局。本文將講解各種常見的佈局及其實現方法,相信一定會對你有所啟發。 ...
  • 地鐵上逛segmentfault看到一篇用純css和SVG來實現的很贊的效果,覺得拿來做一些開場效果動畫應該不錯。 原文地址:https://segmentfault.com/a/1190000010963326 覺得很有趣,正好925快到了,就擼了一個生日快樂的 效果圖如下: 就像是一圈圈螞蟻在它 ...
  • 自我總結 Angular4 開發環境搭建,含 未 翻 牆 解決方案。 一、前期準備(node、npm) 打開 cmd 驗證 node 版本 驗證 npm 版本 現 node 安裝自帶了 npm若安裝老版本,未 翻 牆 可使用淘寶 npm 鏡像 二、node、npm 已配置後全局安裝 Angular ...
  • javaScript 常用技巧總結 1. 徹底屏蔽滑鼠右鍵 2. 取消選取、防止複製 3. 不准粘貼 4. 防止複製 5. IE地址欄前換成自己的圖標 6. 可以在收藏夾中顯示出你的圖標 7. 關閉輸入法 8. 永遠都會帶著框架 9. 防止被人frame 10. 網頁將不能被另存為 11. 查看網頁 ...
  • 1. 定義每個彈射的小球組件( ocicle ) 2. 組件message自定義屬性存放小球初始信息(可修改) 3. 思路 3.1 定時器設置小球每一幀移動 3.2 初始方向:isXtrue為true則小球為橫坐標正方向; isYtrue為true則小球為縱坐標正方向 3.3 每次移動之前獲取小球當 ...
  • 上一篇文章介紹了Bootstrap Table的基本知識點和應用,本文針對上一篇文章中未解決的文件導出問題進行分析,同時介紹BootStrap Table的擴展功能,當行表格數據修改。 1.Bootstrap Bable 全部數據導出分析 在表格導出數據中,發現設置了分頁參數,導出的數據僅為表格載入 ...
  • node.js-v6新版安裝過程 1、Node.js簡介 簡單的說 Node.js 就是運行在服務端的 JavaScript。Node.js 是一個基於 Chrome V8 引擎的 JavaScript 運行環境。Node.js 使用了一個事件驅動、非阻塞式 I/O 的模型,使其輕量又高效。Node ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...