Angularjs的工程化

来源:https://www.cnblogs.com/engpj/archive/2023/01/29/17071932.html
-Advertisement-
Play Games

Angularjs的工程化 AMD規範和CMD規範 為什麼需要模塊化管理工具 在編寫項目時可能需要載入很多js文件,若b.js依賴a.js,且a.js比b.js大很多,那麼瀏覽器會讓b.js等待a.js載入完畢後再去執行b.js里的內容;而即使d.js並不依賴a.js,b.js,c.js,也會等待這 ...


Angularjs的工程化

AMD規範和CMD規範

為什麼需要模塊化管理工具

在編寫項目時可能需要載入很多js文件,若b.js依賴a.js,且a.js比b.js大很多,那麼瀏覽器會讓b.js等待a.js載入完畢後再去執行b.js里的內容;而即使d.js並不依賴a.js,b.js,c.js,也會等待這三個文件均載入完畢才執行,為了讓瀏覽器能夠按需載入,提出了模塊化管理工具。

<script type="text/javascript" src="js/a.js"></script>
<script type="text/javascript" src="js/b.js"></script>
<script type="text/javascript" src="js/c.js"></script>
<script type="text/javascript" src="js/d.js"></script>

AMD規範

全稱為Asynchronous Module Defined,即非同步模塊管理,它通過使用依賴註入等方法完整描述了模塊的定義、依賴關係、引用關係以及載入機制,AngularJS、RequireJS均是符合AMD規範的。

define函數

函數中有三個參數,前兩個參數可以省略,第三個參數是模塊的具體實現本身。 當define函數執行時,它首先會非同步調用第二個參數中列出的依賴模塊,當所有的模塊被載入完成之後,如果第三個參數是一個回調函數則執行,然後告訴系統模塊可用,也通知了依賴於自己的模塊自己已經可用。

define([module-name?], [array-of-dependencies?], [module-factory-or-object]);

其中:
module-name: 模塊標識,可以省略。
array-of-dependencies: 所依賴的模塊,可以省略。
module-factory-or-object: 模塊的實現,或者一個JavaScript對象。

下麵代碼定義了一個alpha模塊,並且依賴於內置的require,exports模塊,以及外部的beta模塊。

define("alpha", ["require", "exports", "beta"], function (require, exports, beta) {
    exports.verb = function() {
    return beta.verb();
    };
});

案例

項目結構

案例
│ test.html

└─js
│ main.js
│ math.js
│ pi.js

└─lib
require.js

項目代碼
test.html
<!DOCTYPE html>
<html>
<head>
	<title>RequireJS學習</title>
</head>
<body>
    <script data-main="js/main.js" src="js/lib/require.js"></script>
    <!-- main.js是主入口文件 -->
</body>
</html>
main.js
requirejs.config({
	baseUrl: 'js',  //所有js程式的根目錄
	paths: {
		//別名
		"math": "math"
	}
});

requirejs(["math"],
	function(math){
alert(math.squre(8));
alert(math.area(10));
	});
alert("hello");
math.js
define(["pi"], function(pi){
	alert("我是math");
	return {
		"squre": function(number){
			return number * number;
		},
		"area": function(r){
			return pi.pi * r *r;
		}
	};
});
pi.js
define({
	"pi": 3.1415926
})
項目說明
  • 在html文件中引包時需要同時指定require.js文件和main.js文件;

  • main.js是主入口文件,只有主入口main.js能用requirejs,其他入口只能用define;

  • math.js用define定義模塊,模塊暴露的API用return返回;

  • main.js中如果有語句不需要依賴別人的語句,可以不寫在回調函數裡面,而現在很少有機會不在回調函數中寫語句,即AMD和CMD越來越像。

CMD規範

全稱為Common Module Defined,即普通模塊管理,其執行過程是懶式的。NodeJS、SeaJS、CommonJS、webpack均是符合CMD規範的。

define函數

define(function(require, exports, module) {  
      // 模塊定義在此  
});

AngularJS的工程化

angular-async-loader

angular-async-loader可以輕鬆解決ReuqireJS和AngularJS之間的粘合問題。angular-async-loader官網

安裝前端依賴

大體思路就是用RequireJS配置AngularJS項目,配置步驟如下。

  • 進入項目文件夾,在命令行視窗執行下列指令,創建bower前端依賴文件。

    bower init
    
  • 創建.bowerrc文件併進行配置(先創建一個空文檔,再在命令行用rename重命名為.bowerrc),配置內容如下,配置後,用bower下載的模塊會生成在assets文件夾下。

    {
    	"directory" : "assets"
    }
    
  • 安裝AngularJS。

    bower install angular --save
    
  • 安裝ui-router。

    bower install angular-ui-router --save
    
  • 安裝RequireJS。

    bower install requirejs --save
    
  • 安裝angular-async-loader。

    bower install angular-async-loader --save
    

書寫三大文件

三大文件為app-routes.js、app.js、bootstrap.js。

在項目根目錄下分別創建ngApp文件夾、app-routes.js、app.js、bootstrap.js、index.html文件。

bootstrap.js

bootstrap.js是RequireJS的入口文件。

require.config({
    baseUrl: '/',
    //別名
    paths: {
        'angular': 'assets/angular/angular.min',
        'angular-ui-router': 'assets/angular-ui-router/release/angular-ui-router.min',
        'angular-async-loader': 'assets/angular-async-loader/dist/angular-async-loader.min'
    },
    //聲明paths中元素暴露的介面和依賴
    shim: {
        'angular': {exports: 'angular'},  //暴露的是angular
        'angular-ui-router': {deps: ['angular']}  //依賴的是angular
    }
});
//核心入口
require(['angular', './app-routes'], function (angular) {
    //當整個文檔就緒之後
    angular.element(document).ready(function () {
        //angular.bootstrap是一個方法,表示啟動angular
        angular.bootstrap(document, ['myapp']);
        //通過類名添加ng-app指令,也可以通過attr來添加
        angular.element(document).find('html').addClass('ng-app');
    });
});

app.js

app.js中創建了app對象。

define(function (require, exports, module) {
    //這是一個CMD規範的模塊,模塊的作用是向外暴露app整體
    //AMD只能向外暴露json形式的API

    //引入依賴
    var angular = require('angular');
    var asyncLoader = require('angular-async-loader');

    require('angular-ui-router');

    //創建app對象,app對象依賴ui.router
    var app = angular.module('app', ['ui.router']);

    // initialze app module for angular-async-loader
    asyncLoader.configure(app);
    //向外暴露
    module.exports = app;
});

app-routes.js

app-routes.js中定義了路由,這裡採用了連續依賴,bootstrap.js依賴app-routes.js,app-routes.js依賴app.js。

define(function (require) {
    //引入app對象
    var app = require('./app');
    //定義路由
    app.config(['$stateProvider', '$urlRouterProvider', function ($stateProvider, $urlRouterProvider) {
        $urlRouterProvider.otherwise('/home');

        $stateProvider
            .state('home', {
                url: '/home',
                template: '<h1>我是首頁!</h1>'
            });
    }]);
});

index.html

index.html是唯一的單頁面,但不表示只存在一個html頁面,其他頁面可作為模板存在,在index.html文件中創建一個ui-view容器,然後用RequireJS語法引用入口文件bootstrap.js。

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8" />
	<title>
		測試系統
	</title>
</head>
<body>
    <ui-view></ui-view>
    
    <script type="text/javascript" src="assets/requirejs/require.js" data-main="bootstrap.js"></script>
</body>
</html>

ngApp

ngApp里可根據場景創建相應文件夾,存放編寫控制器、服務和指令等內容的js文件。此處創建root文件夾和home文件夾。

root
RootCtrl.js
define(function (require) {
    var app = require('app');
    require('./rootService');
    // dynamic load services here or add into dependencies of ui-router state config
    // require('../services/usersService');

    app.controller('RootCtrl', ['$scope', 'rootService', function ($scope, rootService) {
        this.a = rootService.m;
    }]);

});
rootService.js
define(function (require) {
    var app = require('app');

    // dynamic load services here or add into dependencies of ui-router state config
    // require('../services/usersService');

    app.factory("rootService", function () {
        return {
            m : 9
        }
    });

});
template.html
<div>
  <header>
    <h1>我是root的template文件</h1>
    {{rootCtrl.a}}
    <nav>
      <a ui-sref="root.home" ui-sref-active="cur">首頁</a>
    </nav>
  </header>
  <ui-view> </ui-view>
  <footer>我是footer</footer>
</div>

home

HomeCtrl.js
define(function (require) {
    var app = require('app');
    require('jquery');  //var $ = require('jquery');為什麼不用變數接收,因為jquery的原理就是給window對象添加屬性
    require('jquery-ui');
    app.controller('HomeCtrl', [function () {
        this.a = 100;
        $('.box').animate({ 'font-size': 100 }, 1000, function () {
            $(this).css("color", "red");
            $(this).draggable();
        });
    }]);
    
});
template.html
<div>
  <h1>我是首頁。{{homeCtrl.a}}</h1>
  <div class="box">你好</div>
</div>

jquery的引用

法一:

最簡單的方法就是在index.html上引用,這樣可以在全局上使用$函數,缺點是不管頁面是否使用jquery,總是先會載入完畢。

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8" />
    <title>測試系統</title>
    <link rel="stylesheet" href="css/root.css" />
  </head>
  <body>
    <ui-view></ui-view>
    <script
      type="text/javascript"
      src="/assets/jquery/dist/jquery.min.js"
    ></script>
    <script
      type="text/javascript"
      src="/assets/requirejs/require.js"
      data-main="bootstrap.js"
    ></script>
  </body>
</html>

法二:

也可以在bootstrap.js上起一個別名,使用RequireJS載入jquery。

require.config({
    baseUrl: '/',
    //別名
    paths: {
        'angular': 'assets/angular/angular.min',
        'angular-ui-router': 'assets/angular-ui-router/release/angular-ui-router.min',
        'angular-async-loader': 'assets/angular-async-loader/dist/angular-async-loader.min',
        'jquery': 'assets/jquery/dist/jquery.min'
    },
    //聲明paths中元素暴露的介面和依賴
    shim: {
        'angular': {exports: 'angular'},  //暴露的是angular
        'angular-ui-router': { deps: ['angular'] },  //依賴的是angular
        'jquery': {exports: 'jquery'}  //暴露的是jquery
    }
});

然後在需要使用jquery的控制器中寫入如下代碼。

define(function (require) {
    var app = require('app');
    var jquery = require('jquery');
    app.controller('HomeCtrl', [function () {
        jquery('.box').animate({ 'font-size': 100 }, 1000);
    }]);
    
});

若要引入jquery的插件,需要bower下載jquery-ui,然後改變bootstrap.js

require.config({
    baseUrl: '/',
    //別名
    paths: {
        'angular': 'assets/angular/angular.min',
        'angular-ui-router': 'assets/angular-ui-router/release/angular-ui-router.min',
        'angular-async-loader': 'assets/angular-async-loader/dist/angular-async-loader.min',
        'jquery': 'assets/jquery/dist/jquery.min',
        'jquery-ui': 'assets/jquery-ui/jquery-ui.min'
    },
    //聲明paths中元素暴露的介面和依賴
    shim: {
        'angular': {exports: 'angular'},  //暴露的是angular
        'angular-ui-router': { deps: ['angular'] },  //依賴的是angular
        'jquery': {exports: 'jquery'},  //暴露的是jquery
        'jquery-ui': { deps: ['jquery'] },  //依賴的是jquery
    }
});

在控制器中寫入如下代碼。

define(function (require) {
    var app = require('app');
    var $ = require('jquery');
    require('jquery-ui');
    app.controller('HomeCtrl', [function () {
        this.a = 100;
        $('.box').animate({ 'font-size': 100 }, 1000, function () {
            $(this).css("color", "red");
            $(this).draggable();
        });
    }]);
    
});

項目結構

項目
│ app-routes.js
│ app.js
│ bootstrap.js
│ index.html

├─assets
│ ├─angular
│ ├─angular-async-loader
│ ├─angular-ui-router
│ ├─jquery
│ ├─jquery-ui

├─css
│ root.css

└─ngApp
├─home
│ HomeCtrl.js
│ template.html

└─root
RootCtrl.js
rootService.js
template.html

註:在項目中使用ctrl+p,可以定位到目標文件。


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

-Advertisement-
Play Games
更多相關文章
  • xz是什麼 高壓縮率的工具,它使用 LZMA2 壓縮演算法,生成的壓縮文件比傳統使用的 gzip、bzip2 生成的壓縮文件更小, 不過xz也有一個壞處就是壓縮時間比較長,比7z壓縮時間還長一些。不過壓縮主要用於歸檔,不介意的可以忽略。 擅長壓縮文本和日誌文件,針對這塊的壓縮率,是目前我發現效率最高的 ...
  • Docker的常用命令 幫助命令 docker version # docker版本 docker info # 顯示docker的系統信息,包括鏡像和容器的數量 docker [命令] --help # 查看某個具體的命令 鏡像命令 查看下載的所有鏡像 # docker images REPOSI ...
  • 一:背景 1. 講故事 大家都知道資料庫應用程式 它天生需要圍繞著數據文件打轉,諸如包含數據的 .mdf,事務日誌的 .ldf,很多時候深入瞭解這兩類文件的合成原理,差不多對資料庫就能理解一半了,關於 .mdf 的合成前面的文章已經有所介紹,這篇我們來聊一下 .ldf 的一些內部知識,比如 LSN。 ...
  • props props簡單使用 class Person extends React.Component { render() { return ( <ul> <li>姓名:{this.props.name}</li> <li>年齡:{this.props.age}</li> <li>性別:{thi ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 需求描述 目前大多數PC端應用都有配套的移動端APP,如微信,淘寶等,通過使用手機APP上的掃一掃功能去掃頁面二維碼圖片進行登錄,使得用戶登錄操作更方便,安全,快捷。 思路解析 PC 掃碼原理? 掃碼登錄功能涉及到網頁端、伺服器和手機端, ...
  • 一、文件上傳 1、普通文件上傳 JavaScript 可以使用表單提交來實現文件上傳。首先,在 HTML 中創建一個文件輸入框: <input type="file" id="fileInput"> 然後,在 JavaScript 中獲取文件輸入框的引用,併在其上設置事件監聽器,如下所示: var ...
  • 最近在開發幾個網站,為了優化一下前端代碼,就複習一下gulpjs,之前工作gulp用的版本比較老,但是今天看了新的版本,新的語法出了一下,但是為了時間,我決定使用之前gulp3的舊版本,後面發現自己環境的node的版本是最新的,所以不得讓我來瞭解一下gulp的新版本,裡面用到的最多就是commonj ...
  • 一、條件判斷 JavaScript 中有三種方法可以用來進行條件判斷: 1、使用 if-else 語句。這種方法用於在特定條件為 true 時執行一段代碼,否則執行另一段代碼。例如: let a = 5; if (a > 10) { console.log("a is greater than 10 ...
一周排行
    -Advertisement-
    Play Games
  • 1、預覽地址:http://139.155.137.144:9012 2、qq群:801913255 一、前言 隨著網路的發展,企業對於信息系統數據的保密工作愈發重視,不同身份、角色對於數據的訪問許可權都應該大相徑庭。 列如 1、不同登錄人員對一個數據列表的可見度是不一樣的,如數據列、數據行、數據按鈕 ...
  • 前言 上一篇文章寫瞭如何使用RabbitMQ做個簡單的發送郵件項目,然後評論也是比較多,也是準備去學習一下如何確保RabbitMQ的消息可靠性,但是由於時間原因,先來說說設計模式中的簡單工廠模式吧! 在瞭解簡單工廠模式之前,我們要知道C#是一款面向對象的高級程式語言。它有3大特性,封裝、繼承、多態。 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 介紹 Nodify是一個WPF基於節點的編輯器控制項,其中包含一系列節點、連接和連接器組件,旨在簡化構建基於節點的工具的過程 ...
  • 創建一個webapi項目做測試使用。 創建新控制器,搭建一個基礎框架,包括獲取當天日期、wiki的請求地址等 創建一個Http請求幫助類以及方法,用於獲取指定URL的信息 使用http請求訪問指定url,先運行一下,看看返回的內容。內容如圖右邊所示,實際上是一個Json數據。我們主要解析 大事記 部 ...
  • 最近在不少自媒體上看到有關.NET與C#的資訊與評價,感覺大家對.NET與C#還是不太瞭解,尤其是對2016年6月發佈的跨平臺.NET Core 1.0,更是知之甚少。在考慮一番之後,還是決定寫點東西總結一下,也回顧一下.NET的發展歷史。 首先,你沒看錯,.NET是跨平臺的,可以在Windows、 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 添加節點(nodes) 通過上一篇我們已經創建好了編輯器實例現在我們為編輯器添加一個節點 添加model和viewmode ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...
  • 類型檢查和轉換:當你需要檢查對象是否為特定類型,並且希望在同一時間內將其轉換為那個類型時,模式匹配提供了一種更簡潔的方式來完成這一任務,避免了使用傳統的as和is操作符後還需要進行額外的null檢查。 複雜條件邏輯:在處理複雜的條件邏輯時,特別是涉及到多個條件和類型的情況下,使用模式匹配可以使代碼更 ...
  • 在日常開發中,我們經常需要和文件打交道,特別是桌面開發,有時候就會需要載入大批量的文件,而且可能還會存在部分文件缺失的情況,那麼如何才能快速的判斷文件是否存在呢?如果處理不當的,且文件數量比較多的時候,可能會造成卡頓等情況,進而影響程式的使用體驗。今天就以一個簡單的小例子,簡述兩種不同的判斷文件是否... ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...