在AngularJS主页的“创建组件”部分,有这样一个例子:

controller: function($scope, $element) {
  var panes = $scope.panes = [];
  $scope.select = function(pane) {
    angular.forEach(panes, function(pane) {
      pane.selected = false;
    });
    pane.selected = true;
  }
  this.addPane = function(pane) {
    if (panes.length == 0) $scope.select(pane);
    panes.push(pane);
  }
}

注意select方法是如何添加到$scope的,而addPane方法是添加到this的。如果我把它改成$scope。addPane,代码崩溃。

文档说实际上是有区别的,但它没有提到区别是什么:

以前的Angular版本(1.0 RC之前)允许你将此方法与$scope方法互换使用,但现在不再是这样了。在作用域上定义的方法内部this和$scope是可以互换的(angular将this设置为$scope),但在控制器构造函数内部则不能互换。

this和$scope在AngularJS控制器中是如何工作的?


当前回答

$scope有一个不同的this和控制器this。因此,如果你把console.log(this)放在控制器内部,它会给你一个对象(控制器),this.addPane()将addPane方法添加到控制器对象。但是$scope有不同的作用域,它作用域中的所有方法都需要通过$scope. methodname()访问。 this.methodName()在控制器内部意味着在控制器对象中添加方法

$scope.functionName(){
    this.name="Name";
    //or
    $scope.myname="myname"//are same}

将这段代码粘贴到编辑器中,打开控制台查看…

 <!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>this $sope vs controller</title>
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.6.7/angular.min.js"></script>
    <script>
        var app=angular.module("myApp",[]);
app.controller("ctrlExample",function($scope){
          console.log("ctrl 'this'",this);
          //this(object) of controller different then $scope
          $scope.firstName="Andy";
          $scope.lastName="Bot";
          this.nickName="ABot";
          this.controllerMethod=function(){

            console.log("controllerMethod ",this);
          }
          $scope.show=function(){
              console.log("$scope 'this",this);
              //this of $scope
              $scope.message="Welcome User";
          }

        });
</script>
</head>
<body ng-app="myApp" >
<div ng-controller="ctrlExample">
       Comming From $SCOPE :{{firstName}}
       <br><br>
       Comming from $SCOPE:{{lastName}}
       <br><br>
       Should Come From Controller:{{nickName}}
       <p>
            Blank nickName is because nickName is attached to 
           'this' of controller.
       </p>

       <br><br>
       <button ng-click="controllerMethod()">Controller Method</button>

       <br><br>
       <button ng-click="show()">Show</button>
       <p>{{message}}</p>

   </div>

</body>
</html>

其他回答

addPane被赋值给它的原因是<pane>指令。

窗格指令确实需要:'^tabs',它将来自父指令的tabs控制器对象放到link函数中。

addPane被赋值给它,以便窗格链接函数可以看到它。然后在窗格链接函数中,addPane只是选项卡控制器的一个属性,它就是tabscontrollerobject。addPane。因此,pane指令的链接函数可以访问tabs控制器对象,从而访问addPane方法。

我希望我的解释足够清楚。这很难解释。

$scope有一个不同的this和控制器this。因此,如果你把console.log(this)放在控制器内部,它会给你一个对象(控制器),this.addPane()将addPane方法添加到控制器对象。但是$scope有不同的作用域,它作用域中的所有方法都需要通过$scope. methodname()访问。 this.methodName()在控制器内部意味着在控制器对象中添加方法

$scope.functionName(){
    this.name="Name";
    //or
    $scope.myname="myname"//are same}

将这段代码粘贴到编辑器中,打开控制台查看…

 <!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>this $sope vs controller</title>
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.6.7/angular.min.js"></script>
    <script>
        var app=angular.module("myApp",[]);
app.controller("ctrlExample",function($scope){
          console.log("ctrl 'this'",this);
          //this(object) of controller different then $scope
          $scope.firstName="Andy";
          $scope.lastName="Bot";
          this.nickName="ABot";
          this.controllerMethod=function(){

            console.log("controllerMethod ",this);
          }
          $scope.show=function(){
              console.log("$scope 'this",this);
              //this of $scope
              $scope.message="Welcome User";
          }

        });
</script>
</head>
<body ng-app="myApp" >
<div ng-controller="ctrlExample">
       Comming From $SCOPE :{{firstName}}
       <br><br>
       Comming from $SCOPE:{{lastName}}
       <br><br>
       Should Come From Controller:{{nickName}}
       <p>
            Blank nickName is because nickName is attached to 
           'this' of controller.
       </p>

       <br><br>
       <button ng-click="controllerMethod()">Controller Method</button>

       <br><br>
       <button ng-click="show()">Show</button>
       <p>{{message}}</p>

   </div>

</body>
</html>

我建议你阅读下面的帖子: "Controller as"还是"$scope"?

它很好地描述了使用“Controller as”公开变量而不是“$scope”的优点。

我知道你特别问了关于方法而不是变量的问题,但我认为最好坚持一种技术并与之保持一致。

所以在我看来,由于文章中讨论的变量问题,最好只使用“Controller as”技术,并将其应用于方法。

“this和$scope在AngularJS控制器中是如何工作的?”

简短的回答:

this When the controller constructor function is called, this is the controller. When a function defined on a $scope object is called, this is the "scope in effect when the function was called". This may (or may not!) be the $scope that the function is defined on. So, inside the function, this and $scope may not be the same. $scope Every controller has an associated $scope object. A controller (constructor) function is responsible for setting model properties and functions/behaviour on its associated $scope. Only methods defined on this $scope object (and parent scope objects, if prototypical inheritance is in play) are accessible from the HTML/view. E.g., from ng-click, filters, etc.

长一点的回答:

控制器函数是JavaScript的构造函数。当构造函数执行时(例如,当视图加载时),this(即“函数上下文”)被设置为控制器对象。在"tabs"控制器构造函数中,当addPane函数被创建时

this.addPane = function(pane) { ... }

它创建在控制器对象上,而不是在$scope上。视图不能看到addPane函数——它们只能访问在$scope上定义的函数。换句话说,在HTML中,这是行不通的:

<a ng-click="addPane(newPane)">won't work</a>

在"tabs"控制器构造函数执行后,我们得到如下结果:

虚线表示原型继承——一个孤立的范围原型继承自scope。(它并不从HTML中遇到指令的作用域继承。)

现在,pane指令的link函数想要与tabs指令通信(这实际上意味着它需要以某种方式影响tabs隔离$scope)。可以使用事件,但另一种机制是让窗格指令需要选项卡控制器。(似乎没有机制让窗格指令要求选项卡$scope。)

所以,这就引出了一个问题:如果我们只能访问选项卡控制器,我们如何访问选项卡隔离的$scope(这是我们真正想要的)?

Well, the red dotted line is the answer. The addPane() function's "scope" (I'm referring to JavaScript's function scope/closures here) gives the function access to the tabs isolate $scope. I.e., addPane() has access to the "tabs IsolateScope" in the diagram above because of a closure that was created when addPane() was defined. (If we instead defined addPane() on the tabs $scope object, the pane directive would not have access to this function, and hence it would have no way to communicate with the tabs $scope.)

回答你问题的另一部分:$scope如何在控制器中工作?:

在$scope上定义的函数中,这将被设置为“在函数被调用的位置/时间生效的$scope”。假设我们有以下HTML:

<div ng-controller="ParentCtrl">
   <a ng-click="logThisAndScope()">log "this" and $scope</a> - parent scope
   <div ng-controller="ChildCtrl">
      <a ng-click="logThisAndScope()">log "this" and $scope</a> - child scope
   </div>
</div>

而ParentCtrl (only)具有

$scope.logThisAndScope = function() {
    console.log(this, $scope)
}

单击第一个链接将显示this和$scope是相同的,因为“函数被调用时生效的作用域”是与ParentCtrl关联的作用域。

点击第二个链接将显示this和$scope是不一样的,因为“函数被调用时生效的作用域”是与ChildCtrl关联的作用域。这里,这个被设置为ChildCtrl的$scope。在方法内部,$scope仍然是ParentCtrl的$scope。

小提琴

我尽量不在$scope上定义的函数中使用这个,因为这会让人感到困惑,尤其是考虑到ng-repeat、ng-include、ng-switch和指令都可以创建自己的子作用域。

以前的Angular版本(1.0 RC之前)允许你使用这个 可以与$scope方法互换,但这不再是 的情况。在作用域this和$scope上定义的方法内部 可互换(angular将其设置为$scope),否则就不行 在你的控制器构造函数中。

要恢复这种行为(有人知道为什么改变吗?)你可以添加:

return angular.extend($scope, this);

在控制器函数的末尾(假设$scope被注入到这个控制器函数中)。

这有一个很好的效果,可以通过控制器对象访问父作用域,你可以在child中使用require: '^myParentDirective'