指令的执行过程分析。

我们知道指令的执行分两个阶段,一个是compile,一个是link。

我们可以在指令中自定义compile和link。

首先,我们来讲解如何自定义link函数

举个例子:

<!doctype html>
<html ng-app="myModule">
<head>
</head>
<body>
    <hello></hello>
</body>
<script src="angular.js"></script>
<script src="helloAngular.js"></script>
</html>

helloAngular.js代码

var myModule = angular.module("myModule", []);
myModule.directive("hello", function(){
return {
  restrict: "E",
     template: "<div>Hello,Angular</div>",
     replace: true,
link: function(scope, el, attrs, controller){
       el.on("click", function(){
          alert("鼠标点击");
       });
     }
}
});

link方法一般用来对元素进行处理。这里是对元素进行事件的绑定。当用户元素div进行点击时,会弹出鼠标点击的框。

然后,我们来讲解如何自定义compile函数

<!doctype html>
<html ng-app="myModule">
<head>
</head>
<body>
    <div hellos="3">
      <p>hello,Angular!</p>
    </div>
</body>
<script src="angular.js"></script>
<script src="helloAngular.js"></script>
</html>

helloAngular.js代码

var myModule = angular.module("myModule", []);
myModule.directive("hellos", function(){
return {
  restrict: "A",
compile: function(el, attrs, transclude){
       alert("指令编译");
        var tpl = el.children().clone(); //p标签以及内容
        for(var i =0;i<attrs.hellos -1;i++){
          el.append(tpl.clone());
        }
        return function(scope, el, attrs, controller){ //compile必须要return这样的一个函数,这个函数其实就是link函数,在链接阶段,就会执行此link函数。
          alert("指令链接");
        }
     }
}
});

当页面加载进来后,就会弹出指令编译,然后再弹出指令链接的框。

如果,我们在上面的helloAngular.js中,不仅自定义了compile方法,而且也自定义了link方法。比如:

var myModule = angular.module("myModule", []);
myModule.directive("hellos", function(){ //定义一个hellos指令
return {
  restrict: "A",
compile: function(el, attrs, transclude){
       alert("指令编译");
        var tpl = el.children().clone(); //p标签以及内容
        for(var i =0;i<attrs.hellos -1;i++){
          el.append(tpl.clone());
        }
        return function(scope, el, attrs, controller){ //compile必须要return这样的一个函数,这个函数其实就是link函数,在链接阶段,就会执行此link函数。
          alert("指令链接");
        }      }
     link : function(scope, el, attrs, controller){
        alert("自定义的link");
     }
}
});

当页面加载进来时,自定义的link不会弹出来。意思就是说,当存在compile时,不会执行你自定义的link函数,因为angular会把compile返回的函数当做link函数来执行。当不存在compile时,就会执行你自定义的link函数。

compile函数很少被用到,写起来比较麻烦。

compile函数的作用是对指令的模板进行转换,比如,上面的例子中,compile函数就把指令的模板进行了叠加,改变了页面上的DOM结构。

link函数的作用是在模型和视图之间建立关联,同时在元素上绑定事件监听。

scope在链接阶段才会被绑定到元素上,因此在compile中操作scope会报错。

对于同一个指令的多个实例,compile只会执行一次,而link对于指令的每个实例都会执行一次。比如:ng-repeat指令,如果元素的ng-repeat=3,那么ng-repeat指令的compile只会执行一次,而link会执行三次。

对于指令,我们一般只需要自定义link函数就行了。

接下来,我们来讲解使用compile服务,在页面上查找指令并编译指令的过程:

从ng-app开始,递归子层的DOM结构,收集指令,然后在ng-app指令的位置创建$rootScope作用域。比如:上面的代码中,我们注册了一个hellos指令,angular在启动时,就会遍历DOM去查找这个指令。

如果有需要,会为指令生成childScope。

调用每个指令自己的compile函数,生成自己的compositeLinkFn函数。

编译的结果是返回一个publicLinkFn函数。

编译完成之后立即调用生成的publicLinkFn函数。

angular对tree型的数据结构进行双向绑定时,不太理想,性能消耗太大。

加油!

angular核心原理解析3:指令的执行过程的更多相关文章

  1. angular核心原理解析1:angular自启动过程

    angularJS的源代码整体上来说是一个自执行函数,在angularJS加载完成后,就会自动执行了. angular源代码中: angular = window.angular || (window ...

  2. angular核心原理解析2:注入器的创建和使用

    上一课没有讲到创建注入器的方法createInjector. 此方法,会创建两种不同的注入器:第一种叫做providerInjector,第二种叫做instanceInjector.providerI ...

  3. 【算法】(查找你附近的人) GeoHash核心原理解析及代码实现

    本文地址 原文地址 分享提纲: 0. 引子 1. 感性认识GeoHash 2. GeoHash算法的步骤 3. GeoHash Base32编码长度与精度 4. GeoHash算法 5. 使用注意点( ...

  4. Java并发包JUC核心原理解析

    CS-LogN思维导图:记录CS基础 面试题 开源地址:https://github.com/FISHers6/CS-LogN JUC 分类 线程管理 线程池相关类 Executor.Executor ...

  5. ibatis 核心原理解析!

    关注下方公众号,可以在公众号后台回复“博客园”,免费获得作者 Java 知识体系/面试必看资料. 最近查找一个生产问题的原因,需要深入研究 ibatis 框架的源码.虽然最后证明问题的原因与 ibat ...

  6. ibatis 核心原理解析

    最近查找一个生产问题的原因,需要深入研究 ibatis 框架的源码.虽然最后证明问题的原因与 ibatis 无关,但是这个过程加深了对 ibatis 框架原理的理解. 这篇文章主要就来讲讲 ibati ...

  7. Promise核心原理解析

    作者: HerryLo 本文永久有效链接: https://github.com/AttemptWeb...... Promises对象被用于表示一个异步操作的最终完成 (或失败), 及其结果值.主要 ...

  8. 「进阶篇」Vue Router 核心原理解析

    前言 此篇为进阶篇,希望读者有 Vue.js,Vue Router 的使用经验,并对 Vue.js 核心原理有简单了解: 不会大篇幅手撕源码,会贴最核心的源码,对应的官方仓库源码地址会放到超上,可以配 ...

  9. NameServer 核心原理解析

    在之前的文章中,已经把 Broker.Producer 和 Conusmer 的部分源码和核心的机制介绍的差不多了,但是其实 RocketMQ 中还有一个比较关键但是我们平时很容易忽略的组件--Nam ...

随机推荐

  1. 【HTTP请求】、详解

    一.协议介绍 HTTP(HyperText Transfer Protocol,超文本传输协议)是一套计算机通过网络进行通信的规则,使HTTP客户端能够从HTTP服务器端请求到信息和服务,目前的版本号 ...

  2. go 算法

    题目: 请实现一个算法,确定一个字符串的所有字符[是否全都不同].这里我们要求[不允许使用额外的存储结构].给定一个string,请返回一个bool值,true代表所有字符全都不同,false代表存在 ...

  3. 外层div高度不随内层div高度改变的解决办法

    当b1和b2都是float=left时候,a1层的高度不会被b1和b2的高度撑开. <div id=“div1″> <div id=”div2″ style="float: ...

  4. Centos7 下mariadb安装

    1.创建 /etc/yum.repos.d/MariaDB.repo vim  /etc/yum.repos.d/MariaDB.repo 添加如下内容: [mariadb]name=MariaDB ...

  5. Jenkins执行selenium报错unknown error: cannot find Chrome binary

    问题描述:在Pycharm中执行selenium测试用例,可以正常运行, 集成在Jenkins中,构建时,发现构建成功,但是查看Console Output,报错:unknown error: can ...

  6. MySql MyBatis 自动生成主键返回null

    <insert id="insert" parameterType="cn.zno.smse.pojo.UserScan" useGeneratedKey ...

  7. POJ1789 Truck History 2017-04-13 12:02 33人阅读 评论(0) 收藏

    Truck History Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 27335   Accepted: 10634 D ...

  8. 13、容器之间的link

      很多时候我们的业务分为前台和后台,例如:"前台的代码需要连接数据库进行数据操作":但是在写代码的时候我们并不知道后台数据库的地址是什么,所以我们可以docker通过固定的名字来 ...

  9. CGLIB实现动态代理

    JDK动态代理和CGLIB字节码生成的区别? * JDK动态代理只能对实现了接口的类生成代理,而不能针对类 * CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法   因为是继承 ...

  10. Android-Sqlite-OOP方式操作增删改查

    之前写的数据库增删改查,是使用SQL语句来实现的,Google 就为Android开发人员考虑,就算不会SQL语句也能实现增删改查,所以就有了OOP面向对象的增删改查方式 其实这种OOP面向对象的增删 ...