一. 指令模板选项有complie和link两个字段,两者之间存在如下关系:

  • 当compile字段存在时,link字段将被忽略,compile函数的返回值将作为link字段。
  • 当compile不存在,link字段存在时,angular通过这样directive.compile = valueFn(directive.link);包装一层,使用用户定义的link字段。
  • 当compile和link同事存在时,link被忽略,使用compile函数的返回值将作为link字段,
而link分为preLink和postLink两个阶段,从link字段或者compile函数的返回值来看:
  • 如果是函数,那么这样的link,会被认为是postLink。
  • 如果是对象,那么link.pre作为preLink函数,link.post作为postLink函数 (如下)
app.directive('myDirective', function () {
return {
compile: function () {
return {
pre: function () {
console.log('preLink');
},
post: function () {
console.log('postLink');
}
}
}
}
});

  

angular 启动后,会从ng-app所在元素,递归遍历所有子元素,查找出所有的指令(包括指令模板中的指令),对指令进行编译和连接

编译会执行指令的compile(即使有多个指令实例,compile也只执行一次),·链接会执行指令的link(prelink,postlink)(有多少个指令实例,就执行几次)

二. compile,prelink,postlink的执行顺序

对于一个指令而言,首先会执行compile,然后执行prelink,最后执行postlink

对于所有的指令而言,执行顺序如下

compile的执行顺序

编译过程会按照从外到内(从父到子),从上到下(兄弟节点)以及priority的依次执行所有指令的compile,

prelink和postlink的执行顺序

所有指令的compile都执行完成后,会执行链接过程

链接过程会首先按照从外到内(从父到子)的顺序执行指令的prelink,然后再按照从内到外(从子到父)的顺序执行指令的postlink

从ng-app开始,会依次按照以下顺序执行prelink和postlink

1.如果有子节点指令,那么执行完本节点指令的prelink后,就会执行子节点指令的prelink,

2.如果没有子节点指令,那么会执行完本节点指令的prelink后,就会执行本节点指令的postlink,

3.如果有兄弟节点指令,那么执行完本节点指令的postlink后,就会去执行兄弟节点指令的prelink,

4.如果没有兄弟节点指令,那么执行完本节点指令的postlink后,就会去执行父节点指令的postlink,

以此类推,直到所有的指令的prelink和postlink执行完毕

测试链接:http://plnkr.co/edit/KtMs0H1pBsrOmXrFh9nf?p=preview

事例

html代码

var app = angular.module('plunker', []);

function createDirective(name){
return function(){
return {
restrict: 'E',
controller: function () {
console.log(name + ': controller => ');
},
compile: function(tElem, tAttrs){
console.log(name + ': compile => ' );
return {
pre: function(scope, iElem, iAttrs){
console.log(name + ': pre link => ' );
},
post: function(scope, iElem, iAttrs){
console.log(name + ': post link => ' );
}
}
}
}
}
} app.directive('levelOne', createDirective('levelOne'));
app.directive('levelTwo1', createDirective('levelTwo1'));
app.directive('levelThree1', createDirective('levelThree1'));
app.directive('levelFour1', createDirective('levelFour1'));
app.directive('levelTwo2', createDirective('levelTwo2'));
app.directive('levelThree2', createDirective('levelThree2'));
app.directive('levelFour2', createDirective('levelFour2'));

  

  结果

app.js:11 levelOne: compile =>
app.js:11 levelTwo1: compile =>
app.js:11 levelThree1: compile =>
app.js:11 levelFour1: compile =>
app.js:11 levelTwo2: compile =>
app.js:11 levelThree2: compile =>
app.js:11 levelFour2: compile =>
app.js:8 levelOne: controller =>
app.js:14 levelOne: pre link =>
app.js:8 levelTwo1: controller =>
app.js:14 levelTwo1: pre link =>
app.js:8 levelThree1: controller =>
app.js:14 levelThree1: pre link =>
app.js:17 levelThree1: post link =>
app.js:8 levelFour1: controller =>
app.js:14 levelFour1: pre link =>
app.js:17 levelFour1: post link =>
app.js:17 levelTwo1: post link =>
app.js:8 levelTwo2: controller =>
app.js:14 levelTwo2: pre link =>
app.js:8 levelThree2: controller =>
app.js:14 levelThree2: pre link =>
app.js:17 levelThree2: post link =>
app.js:8 levelFour2: controller =>
app.js:14 levelFour2: pre link =>
app.js:17 levelFour2: post link =>
app.js:17 levelTwo2: post link =>
app.js:17 levelOne: post link =>

  

  

可以看出:
  1. 所有的指令都是先compile,然后preLink,然后postLink。
  2. 节点指令的preLink是在所有子节点指令preLink,postLink之前,所以一般这里就可以通过scope给子节点传递一定的信息。
  3. 节点指令的postLink是在所有子节点指令preLink,postLink完毕之后,也就意味着,当父节点指令执行postLink时,子节点postLink已经都完成了,此时子dom树已经稳定,所以我们大部分dom操作,访问子节点都在这个阶段。
  4. 指令在link的过程,其实是一个深度优先遍历的过程,postLink的执行其实是一个回溯的过程。
  5. 节点上的可能有若干指令,在搜集的时候就会按一定顺序排列(通过byPriority排序),执行的时候,preLinks是正序执行,而postLinks则是倒序执行。

 三. compile,prelink,postlink的作用

Compile 函数

element以及相关的属性是做为参数传递给compile函数的,不过这时候scope是不能用的

使用compile函数可以改变原始的dom(template element),在ng创建原始dom实例以及创建scope实例之前.

可以应用于当需要生成多个element实例,只有一个template element的情况,ng-repeat就是一个最好的例子,它就在是compile函数阶段改变原始的dom生成多个原始dom节点,然后每个又生成element实例.因为compile只会运行一次,所以当你需要生成多个element实例的时候是可以提高性能的.

Pre-link 函数

preLink是在所有子节点指令preLink,postLink之前,所以一般这里就可以通过scope给子节点传递一定的信息.

scope对象以及element实例将会做为参数传递给pre-link函数:

Post-link 函数

postLink是在所有子节点指令preLink,postLink完毕之后,也就意味着,当父节点指令执行postLink时,子节点postLink已经都完成了,

此时子dom树已经稳定,所以我们大部分dom操作,访问子节点都在这个阶段。

可以再次函数做事件的绑定

这就是被认为是最安全以及默认的编写业务逻辑代码的原因.

scope实例以及element实例做为参数传递给post-link函数:

 controller

之前已经讲过compile,prelink 和 postlink的关系,执行顺序,以及作用,这里在追加controller

controler是在compile之后,prelink之前执行,可以在controller中定义一些数据供prelink或postlink使用

或者定义一些方法供其他的指令调用。

可以说controller提供一些外部接口,供其他指令调用,link只处理当前指令。

指令的controller可以通过 指令名+Ctrl 的方式注入到其他指令(指令的第四个参数)

参考: https://blog.csdn.net/qq_28506819/article/details/72598457

angular指令的compile,prelink 和 postlink以及controller的更多相关文章

  1. angular指令中的preLink函数和postLink函数

    指令模板选项有complie和link两个字段,两者之间存在如下关系: 当compile字段存在时,link字段将被忽略,compile函数的返回值将作为link字段. 当compile不存在,lin ...

  2. directive(指令里的)的compile,pre-link,post-link,link,transclude

    The nitty-gritty of compile and link functions inside AngularJS directives  The nitty-gritty of comp ...

  3. Angular1.x directive(指令里的)的compile,pre-link,post-link,link,transclude

    The nitty-gritty of compile and link functions inside AngularJS directives  The nitty-gritty of comp ...

  4. angular学习笔记(三十)-指令(7)-compile和link(2)

    继续上一篇:angular学习笔记(三十)-指令(7)-compile和link(1) 上一篇讲了compile函数的基本概念,接下来详细讲解compile和link的执行顺序. 看一段三个指令嵌套的 ...

  5. ng directive compile pre-link post-link

    原文链接: http://www.jb51.net/article/58229.htm 1.ng在link之前编译所有的指令,然后link又分为 pre-link 与 post-link阶段compi ...

  6. angular中的compile和link函数

    angular中的compile和link函数 前言 这篇文章,我们将通过一个实例来了解 Angular 的 directives (指令)是如何处理的.Angular 是如何在 HTML 中找到这些 ...

  7. Angular指令1

    Angular的指令 也就是directive,其实就是一WebComponent,以前端的眼光来看,好象很复杂,但是以后端的眼光来看,还是非常简单的.其实就是一个中等水平的类. var myModu ...

  8. 如何编写Angular指令

    [20140917]Angular:如何编写一个指令 *:first-child { margin-top: 0 !important; } body>*:last-child { margin ...

  9. angular指令之complie和link不得不说的故事

    angular指令比较晦涩难懂的就是complie和link字段了,什么时候该用complie?什么时候该用link?总是很难分别清楚.当理解了指令的真正编译原理的时候,就会发现这相当的简单. ng怎 ...

随机推荐

  1. Java项目部署服务器操作

    有 2个工具需要下载,我使用的是 xshell(操作Linux命令),xftp5(操作文件传输) 需要知道服务器 ip ,账号,密码 xshell连接时,协议选择SSH连接,其他正常输入. xftp5 ...

  2. [UE4]通过使用Set TimerByFunctionName来实现反射机制

  3. a++ 与 ++a 的运算

    var a=5: b=a++和b=++a的区别: 前者是先赋值,再自加,即b=a:a=a+1: //结果b=5,a=6 后者是先自加,再赋值,即a=a+1;b=a;  //结果a=6,b=6

  4. 屏蔽windows快捷键的方法

    using System; using System.Collections.Generic; using System.Text; using System.Windows.Forms; using ...

  5. CF865D Buy Low Sell High

    /* 贪心来选择, 如果能找到比当前小的, 就用最小的来更新当前的 优先队列即可 */ #include<cstdio> #include<algorithm> #includ ...

  6. elasticsearch License 到期后更新步骤

    ELK下载安装后有一个月试用期,到期后需要更新License,且每个License有效期为 1 年,License过期前10天里相关log会一直出现,提醒用户更新,支持实时更新无需重启服务. 步骤: ...

  7. vscode 正则 计算代码全部有效行数;清除文件空行

    清除空行: 正则表达式:^\s*(?=\r?$)\n 计算全部代码行数: 正则表达式:b*[^:b#/]+.*

  8. WPF开发ArcGis系统时的异常信息: ArcGIS product not specified. You must first bind to an ArcGIS version prior to using any ArcGIS components.

    “System.Runtime.InteropServices.COMException”类型的未经处理的异常在 Arcgis_Test.exe 中发生 其他信息: ArcGIS product no ...

  9. 零基础学习python_列表和元组(10-13课)

    一时兴起今天又回过头来补一下列表和元组,先来说说列表哈,列表其实是python最经常用到的数据类型了,不仅经常用还很强大呢,这个跟C语言里面的数组是类似的,列表当然也可以增删改查,不过我可没打算用之前 ...

  10. [java] 求2个集合的交 差 并集

    要求2个集合的交 差 并集. set集合,如下 import java.util.HashSet; import java.util.Set; public class SetTest { publi ...