admin管理员组文章数量:1344238
I have made a directive that, when clicked, creates a dialog that is appended to the body using jQuery. The problem is that when the dialog is closed, the scopes are never properly cleaned up. As shown in the picture below 167 ChildScopes are preserved. Which matches the amount of items in the dialog which includes the ng-repeat directive.
I attempted to create an extremely simple version of the scenario on Plnkr. To my surprise the scopes ARE in fact being removed on each close in the Plnkr. So something, somewhere in production is causing the scopes to stay alive even after $destroy
has been called.
link: ($scope, $element, $attr) ->
$element.on 'click', () ->
$scope.$apply () ->
child = $scope.$new()
template = """<span ng-controller="ListCtrl">...List dialog things...</span>"""
piledTemplate = $pile(template)(child)
container = containers.createDialogContainer($element)
container.append(piledTemplate)
#cleanup
$scope.closeWidget = () ->
container.trigger("container_close")
return
container.on "container_close", ()->
child.$apply () ->
child.$destroy()
return
So here is my question:
What can cause a scope to stay alive even after $destroy has been called, triggered and garbage collection performed?
For obvious reasons I cannot show you our production code. However the directive in the Plnkr matches the one im debugging sufficiently.
I have made a directive that, when clicked, creates a dialog that is appended to the body using jQuery. The problem is that when the dialog is closed, the scopes are never properly cleaned up. As shown in the picture below 167 ChildScopes are preserved. Which matches the amount of items in the dialog which includes the ng-repeat directive.
I attempted to create an extremely simple version of the scenario on Plnkr. To my surprise the scopes ARE in fact being removed on each close in the Plnkr. So something, somewhere in production is causing the scopes to stay alive even after $destroy
has been called.
link: ($scope, $element, $attr) ->
$element.on 'click', () ->
$scope.$apply () ->
child = $scope.$new()
template = """<span ng-controller="ListCtrl">...List dialog things...</span>"""
piledTemplate = $pile(template)(child)
container = containers.createDialogContainer($element)
container.append(piledTemplate)
#cleanup
$scope.closeWidget = () ->
container.trigger("container_close")
return
container.on "container_close", ()->
child.$apply () ->
child.$destroy()
return
So here is my question:
What can cause a scope to stay alive even after $destroy has been called, triggered and garbage collection performed?
For obvious reasons I cannot show you our production code. However the directive in the Plnkr matches the one im debugging sufficiently.
Share Improve this question edited Feb 19, 2014 at 7:53 CodePrimate asked Feb 18, 2014 at 13:46 CodePrimateCodePrimate 6,66613 gold badges50 silver badges86 bronze badges 11- 1 "To my surprise the scopes ARE in fact being removed on each close in the Plnkr. So something, somewhere in production is causing the scopes to stay alive even after $destroy has been called." --> If we can't reproduce the issue using the information you've given us, we can't solve your issue without psychic debugging. – George Stocker Commented Feb 18, 2014 at 14:43
- 1 @GeorgeStocker the thing is I'm not looking for a specific solution. I'm looking for possible answers to the question: "what can cause a scope to stay alive?". I realize that with the information given it would be impossible to pinpoint where exactly my problem lies. I was however hoping that someone out there had experienced issues with "cleaning up" angular scopes in the past and could shed some light on how the problem was resolved. – CodePrimate Commented Feb 18, 2014 at 14:47
- Destroy event as per documentation: "Removes the current scope (and all of its children) from the parent scope. Removal implies that calls to $digest() will no longer propagate to the current scope and its children. Removal also implies that the current scope is eligible for garbage collection.". That it is eligible for garbage collection, doesn't mean it is deleted yet. – pedromarce Commented Feb 18, 2014 at 15:04
- 1 I think this is one of those times when the debug strategy has to be cutting out half the code, seeing if it still happens, if so, cut out another half the code, etc. until you have isolated the problem to a small example. Then if you still have a question, update your question with just that portion of the code. – Katie Kilian Commented Feb 18, 2014 at 15:08
- @GeorgeStocker I think it is a genuine question, and it actually has an answer, he expects scopes to be gone by the time destroy event happens, but it doesn't, which is not unexpected, actually plunkr in this case I think it is unnecessary. – pedromarce Commented Feb 18, 2014 at 15:12
3 Answers
Reset to default 5In general, a scope (or any other JS object) can not be cleaned up by the GC if it is still accessible by another JS object.
In practice, in an Angular project that also uses JQuery, this is most likely caused by:
- an Angular service, controller, scope or some other object still having a reference to your scope object
- a reference to your object still exists through a DOM element, probably through an event listener. The DOM element might not be GC-able itself because it is still in the cache of JQuery
Your example, for instance, creates a memory leak.
From your code:
$scope.removeDialog = () ->
console.log "closing"
child.$destroy()
$('.listshell').remove()
return
You do not set child
to null
so $scope.removeDialog
still has access to the scope object referenced by the child
variable. Therefore, this object can not be GC'ed.
NB: It seems to me that it would be more appropriate to put removeDialog
on the child scope. Now your example only works because the child scope is not isolated.
Closure functions can cause a functions Activation object to stay alive even after the scope has been "destroyed". For instance you might have inner functions still referencing variable objects in the functions whose scope you are trying to destroy.
nullifying the reference variables would be the best option as opposed to delete
The only thing I can think of is that if you somewhere have a global function or function outside of the controller or directive that references a method inside of the scope for the directive it will infact keep that scope alive for the duration of the application.
本文标签: javascriptWhy are scopes not destroyed even after destroy is triggeredStack Overflow
版权声明:本文标题:javascript - Why are scopes not destroyed even after $destroy is triggered? - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743745566a2531691.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论