post-link的用法与区别." />
Angularjs   发布时间:2022-04-20  发布网站:大佬教程  code.js-code.com
大佬教程收集整理的这篇文章主要介绍了AngularJS指令中的compile与link函数解析大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。

通常大家在使用ng中的指令的时候,用的链接函数最多的是link属性,下面这篇文章将告诉大家complie,pre-link,244)">post-link的用法与区别.

原文地址

angularjs里的指令非常神奇,允许你创建非常语义化以及高度重用的组件,可以理解为web components的先驱者.

网上已经有很多介绍怎么使用指令的文章以及相关书籍,相互比较的话,很少有介绍compilelink的区别,更别说pre-linkpost-link了.

大部分教程只是简单的说下compile会在ng内部用到,而且建议大家只用是这样的

这是非常不幸的,因为正确的理解这些函数的区别会提高你对ng内部运行机理的理解,有助于你开发更好的自定义指令.

所以跟着我一起来看下面的内容一步步的去了解这些函数是什么以及它们应该在什么时候用到

@H_874_51@ NG中是怎么样处理指令的

开始分析之前,先让我们看看ng中是怎么样处理指令的.

当浏览器渲染一个页面时,本质上是读html标识,然后建立dom节点,当dom树创建完毕之后广播一个事件给我们.

当你在页面中使用script标签加载ng应用程序代码时,244)">ng监听上面的dom完成事件,查找带有ng-app属性的元素.

当找到这样的元素之后,244)">ng开始处理dom以这个元素的起点,所以假如ng-app添加html元素上,则ng就会从html元素开始处理dom.

从这个起点开始,244)">ng开始递归查找所有子元素里面,符合应用程序里定义好的指令规则.

ng怎样处理指令其实是依赖于它定义时的对象属性的,你可以定义一个compile或者一个link函数,或者用pre-linkpost-link函数来代替link.

所以这些函数的区别呢?为什么要使用它?以及什么时候使用它呢?

带着这些问题跟着我一步一步来解答这些迷团吧

@H_874_51@ 一段代码

为了解释这些函数的区别,下面我将使用一个简单易懂的例子

看看下面一段html标签代码

<level-one>
        <level-two>
            <level-three>
                Hello 
            </level-three>
        </level-two>
    </level-one>@H_673_136@ 
   

然后是一段js代码

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

    function createDirective(name){
      return function(){
        return {
          reStrict: 'E',
          compile: function(tElem, tAttrs){
            console.log(name + ': compile');
            return {
              prefunction(scope, iElem, iAttrs){
                console.log(name ': pre link');
              },
              post': post link');
              }
            }
          }
        }
      }
    }

    app.directive('levelOne', createDirective('levelOne'));
    app.directive('levelTwo',33)">'levelTwo'));
    app.directive('levelThree',33)">'levelThree'));@H_673_136@ 
   

结果非常简单:让ng来处理三个嵌套指令,并且每个指令都有自己的complile,244)">post-link函数,每个函数都会在控制台里打印一行东西来标识自己.

这个例子能够让我们简单的了解到ng在处理指令时,内部的流程

@H_874_51@ 代码输出

下面是一个在控制台输出结果的截图

如果想自己试一下这个例子的话,请点击this plnkr,然后在控制台查看结果.

@H_874_51@ 分析代码

一个要注意的是这些函数调用顺序:

// COMPILE PHASE
    // levelOne: compile function is called
    // levelTwo: compile function is called
    // levelThree: compile function is called

    // PRE-LINK PHASE
    // levelOne: pre link function is called
    // levelTwo: pre link function is called
    // levelThree: pre link function is called

    // POST-LINK PHASE (Notice the reverse order)
    // levelThree: post link function is called
    // levelTwo: post link function is called
    // levelOne: post link function is called@H_673_136@ 
   

这个例子清晰的显示出了nglink之前编译所有的指令,然后link要又分为了post-link阶段.

注意下,244)">pre-link的执行顺序是依次执行的,但是post-link正好相反.

所以上面已经明确标识出了不同的阶段,244)">pre-link有什么区别呢,都是相同的执行顺序,为什么还要分成两个不同的函数呢?

@H_874_51@ DOM

为了挖的更深一点,让我们简单的修改一下上面的代码,它也会在各个函数里打印参数列表中的element变量

function createDirective(name){ 
      @H_35_419@': compile => ' + tElem.html());
            @H_35_419@': pre link => ' + iElem.html());
              },33)">': post link => ' + iElem.html());
              }
            }
          }
        }
      }
    }

    app.directive('levelThree'));@H_673_136@ 
   

注意下console.log里的输出,除了输出原始的html标记基本没别的改变.

这个应该能够加深我们对于这些函数上下文的理解.

再次运行代码看看

@H_874_51@ 输出

下面是一个在控制台输出结果的截图

假如你还想自己运行看看@L_618_61@,可以点击输出结果.

@H_874_51@ 观察

输出dom的结果可以暴露一些有趣的事情:dom内容pre-link两个函数中是不一样的

@H_874_51@ 所以发生了什么呢?

Compile

我们已经知道当ng发现dom构建完成时就开始处理dom.

所以当ng在遍历dom的时候,碰到level-one元素,从它的定义那里了解到,要执行一些必要的函数

因为compile函数定义在level-one指令的指令对象里,所以它会被调用并传递一个element对象作为它的参数

如果你仔细观察,就会看到,浏览器创建这个element对象时,仍然是最原始的html标记

  • ng中,原始dom通常用来标识template element,所以我在定义compile函数参数时就用到了tElem名字,这个变量指向的就是template element.

一旦运行levelone指令中的compile函数,244)">ng就会递归深度遍历它的level-twolevel-three上面重复这些操作.

Post-link

深入了解pre-link函数之前,让我们来看看post-link函数.

  • 如果你在定义指令的时候只使用了一个ng会把这个函数当成post-link来处理,因此我们要先讨论这个函数

ng遍历完所有的dom并运行完所有的compile函数之后,就反向调用相关联的post-link函数.

dom现在开始反向,并执行调用看起来有点奇怪,其实这样做是非常有意义的.

当运行包含子指令的指令post-link时,反向的post-link规则可以保证它的子指令的post-link是已经运行过的.

所以,当运行level-one指令的post-link函数的时候,我们能够保证level-twolevel-threepost-link其实都已经运行过了.

这就是为什么人们都认为post-link是最安全或者认的写业务逻辑的地方.

但是为什么这里的elementcompile里的又不同呢?

一旦ng调用过指令的一个template elementelement实例对象,并且为它提供一个scope对象,这个scope有可能是新实例,也有可能是已经存在,可能是个子scope,也有可能是独立的scope属性

所以当linking发生时,这个实例element以及scope对象已经是可用的了,并且被ng作为参数传递到post-link函数的参数列表中去.

  • 我个人总是使用iElem名称来定义一个link函数的参数,并且它是指向element实例的

所以post-link(pre-link)函数element参数对象是一个element实例而不是一个template element.

所以上面例子里的输出是不同的

Pre-link

当写了一个674@post-link函数是已经执行过的.

在大部分的情况下,它都可以做的更好,因此通常我们都会使用它来编写指令代码.

然而,244)">ng为我们提供了一个附加的hook机制,那就是pre-link函数,它能够保证在执行所有子指令的post-link函数之前.运行一些别的代码.

这句话是值得反复推敲的

pre-link函数能够保证在element实例上以及它的所有子指令的post-link运行之前执行.

所以它使的post-link函数反向执行是相当有意义的,它自己是原始的顺序执行pre-link函数

这也意为着pre-link函数运行在它所有子指令的一个元素的pre-link函数能够保证是运行在它所有的子指令的post-linkpre-link运行之前执行的.见下图

回顾

如果我们回头看看上面原始的输出,就能清楚的认出到底发生了什么:

// HERE THE ELEMENTS ARE STILL THE ORIGINAL TEMPLATE ELEMENTS

    // levelOne: compile function is called on original DOM
    // levelTwo: compile function is called on original DOM
    // levelThree: compile function is called on original DOM

    // AS OF HERE,THE ELEMENTS HAVE BEEN INSTANTIATED AND
    // ARE BOUND TO A SCOPE
    // (E.G. NG-REPEAT WOULD HAVE MULTIPLE INSTANCES)

    // levelOne: pre link function is called on element instance
    // levelTwo: pre link function is called on element instance
    // levelThree: pre link function is called on element instance

    // levelThree: post link function is called on element instance
    // levelTwo: post link function is called on element instance
    // levelOne: post link function is called on element instance@H_673_136@ 
   
@H_874_51@ 概要

回顾上面的分析我们可以描述一下这些函数的区别以及使用情况:

Compile 函数

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

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

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

下面是函数样子

@H_585_801@/**
@H_585_801@ * Compile function
@H_585_801@ * 
@H_585_801@ * @param tElem - template element
@H_585_801@ * @param tAttrs - attributes of the template element
@H_585_801@ */
    :rgb(0, tAttrs){

        // ...

    };@H_673_136@ 
   

Pre-link 函数

使用pre-link函数可以运行一些业务代码ng执行完post-link函数将要执行之前.

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

下面是函数样子

 * Pre-link function
@H_585_801@ * @param scope - scope associated with this istance
@H_585_801@ * @param iElem - instance element
@H_585_801@ * @param iAttrs - attributes of the instance element
:rgb(64, iAttrs){

         Post-link 函数 
   

使用post-link函数来执行业务逻辑,在这个阶段,它已经知道它所有的子指令已经编译完成并且pre-link以及post-link函数已经执行完成.

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

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

下面是函数样子

 * Post-link function@H_673_136@
@H_585_801@// ... }; @H_874_51@ 总结

现在你应该对compile,244)">post-link@H_673_136@

这此函数间的区别有了清晰的认识了吧.

如果还没有的话,并且你是一个认真的ng开发者,那么我强烈建议你重新把这篇文章读一读直到你了解为止

理解这些概念非常重要,能够帮助你理解ng原生指令的工作原理,也能帮你优化你自己的自定义指令.

如果还有问题的话,欢迎大家在下面评论加上你的问题

以后还会接着分析关于指令里的其它两个问题:

  • 指令使用transclusion属性是怎么工作的?

  • 指令的controller函数是怎么关联的

最后,如果发现本文哪里有不对的,请及时给我发评论

谢谢

本图文内容来源于网友网络收集整理提供,作为学习参使用,版权属于原作者。

猜你在找的Angularjs相关文章

angular.js实现数据双向通信的原理angular的核心特性有:MVVM、模块化、依赖注入、自动化双向数据绑定、语义标签等。1、AngularJS的scopes对象AngularJS的scopes对象,是一般的javascript对象,可以在他们上面绑定属性和其他对象,也可以添加一些功能,用于观察数据结构上的变化。观察功能都由dirty-checking来实现,并且都在一个digest循环中...
AngularJS 是一个JavaScript 框架。它可通过 注:建议把脚本放在 元素的底部。这会提高网页加载速度,因为 HTML 加载不受制于脚本加载。AngularJS 扩展了 HTMLAngu
AngularJS 通过被称为指令的新属性来扩展 HTML。AngularJS 指令AngularJS 指令是扩展的 HTML 属性,带有前缀ng-。ng-app指令初始化一个 AngularJS 应
AngularJS 使用表达式把数据绑定到 HTML。AngularJS 表达式AngularJS 表达式写在双大括号内:{{ expression }}。AngularJS 表达式把数据绑定到 HT
$http是 AngularJS 中的一个核心服务,用于读取远程服务器的数据。读取 JSON 文件下是存储在web服务器上的 JSON 文件:{ &quot;records&quot;: [ { &q
AngularJS 控制器控制AngularJS 应用程序的数据。AngularJS 控制器是常规的JavaScript 对象。AngularJS 控制器AngularJS 应用程序被控制器控制。ng
keyup适用于文本框的数据输入和同步,以及数据的获取;keydown 与 keypress更适用于通过键盘控制页面功能的实现(如回车事件)
angularJs模板缓存的清除,包括传统的 HTML标签设置清除缓存,以及angularJs的一些配置清除,和angularJs的路由切换操作清除
@H_163_1058@

大佬总结

以上是大佬教程为你收集整理的AngularJS指令中的compile与link函数解析全部内容,希望文章能够帮你解决AngularJS指令中的compile与link函数解析所遇到的程序开发问题。

如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。