1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
|
---
title: Gerenciamento de Memória
slug: Web/JavaScript/Memory_Management
tags:
- coletor
- memória(2)
translation_of: Web/JavaScript/Memory_Management
---
<div>{{JsSidebar("Advanced")}}</div>
<h2 id="Introdução">Introdução</h2>
<p>Linguagens de baixo nível, como C, tem primitivas de gerenciamento de memória de baixo nível como <code>malloc()</code> e <code>free()</code>. Em contrapartida, os valores do JavaScript são alocados quando coisas (objetos, strings, etc.) são criadas e "automaticamente" liberadas quando não são mais usadas. Este último processo se chama <em>garbage collection</em>. Facilmente se torna uma fonte de confusão e dá a impressão aos desenvolvedores JavaScript (e outras linguagens de alto nível) que eles não precisam se preocupar com o consumo de memória. Isto é um erro.</p>
<h2 id="Ciclo_de_vida_da_memória">Ciclo de vida da memória</h2>
<p>Independentemente da linguagem de programação, o ciclo de vida da memória é praticamente sempre o mesmo:</p>
<ol>
<li>Alocar a memória que você precisa</li>
<li>Utilizar a memória alocada (ler, escrever)</li>
<li>Liberar a memória alocada quando não é mais necessária</li>
</ol>
<p>A primeira e a segunda parte são explícitas em todas as linguagens. A última parte é explicita em linguagens de baixo nível, porém implícito em linguagens de alto nível como JavaScript.</p>
<h3 id="Alocação_no_JavaScript">Alocação no JavaScript</h3>
<h4 id="Inicialização_de_valor">Inicialização de valor</h4>
<p>A fim de não incomodar o programador com alocações, o JavaScript faz isso com os valores conforme são declarados.</p>
<pre class="brush: js">var n = 123; // aloca memória para um número
var s = "azerty"; // aloca memória para uma string
var o = {
a: 1,
b: null
}; // aloca memória para um objeto e seus valores contidos
// (assim como o objeto) aloca memória para o vetor e
// seus valores contidos
var a = [1, null, "abra"];
function f(a) {
return a + 2;
} // aloca uma função (que é um objeto que pode ser chamado)
// expressões de funções também alocam um objeto
someElement.addEventListener('click', function(){
someElement.style.backgroundColor = 'blue';
}, false);
</pre>
<h4 id="Alocação_via_chamada_de_uma_função">Alocação via chamada de uma função</h4>
<p>Algumas funções quando chamadas resultam na alocação de um objeto.</p>
<pre class="brush: js">var d = new Date();
// aloca um elemento do DOM
var e = document.createElement('div');
</pre>
<p>Alguns métodos alocam novos valores ou objetos:</p>
<pre class="brush: js">var s = "azerty";
var s2 = s.substr(0, 3); // s2 é uma nova string
// Como as strings são valores imutáveis,
// o JavaScript pode decidir não alocar memória,
// mas apenas armazenar o intervalo [0, 3].
var a = ["ouais ouais", "nan nan"];
var a2 = ["generation", "nan nan"];
var a3 = a.concat(a2);
// novo vetor com 4 elementos sendo
// a concatenação dos elementos a e a2
</pre>
<h3 id="Utilização_de_valores">Utilização de valores</h3>
<p>A utilização de valores basicamente significa leitura e escrita em memória alocada. Isto pode ser feito ao ler ou escrever o valor de uma variável ou a propriedade de um objeto ou até mesmo ao passar um argumento para uma função.</p>
<h3 id="Libere_quando_a_memória_não_for_mais_necessária">Libere quando a memória não for mais necessária</h3>
<p>A maioria dos problemas relacionados ao gerenciamento de memória aparecem nesta fase. A tarefa mais complicada aqui é descobrir quando "a memória alocada não é mais necessária". Geralmente exige que o desenvolvedor determine a onde no programa tal pedaço da memória não é mais necessária e liberá-la.</p>
<p>Linguagens de alto nível vêm com um pedaço de software chamado "garbage collector" (coletor de sujeira), cujo trabalho é monitorar a alocação de memória a fim de descobrir quando um pedaço de código não é mais necessário e neste caso, automaticamente liberá-lo. Este processo é algo aproximado já que, em geral, saber se um pedaço de memória é necessário é algo <a class="external" href="http://pt.wikipedia.org/wiki/Decidibilidade">indecidível</a> (que não pode ser resolvido através de um algoritmo).</p>
<h2 id="Garbage_collection">Garbage collection</h2>
<p>Como foi mencionado acima, em geral o problema de automaticamente descobrir se a memória "não é mais necessária" é indecidível. Como consequência, os <em>garbage collections</em> implementam uma limitação de uma solução ao problema em geral. Esta seção irá explicar os conceitos necessários para entender os principais algoritmos de garbage collection e suas limitações.</p>
<h3 id="Referências">Referências</h3>
<p>O principal conceito de algoritmos do garbage collection depende do conceito de <em>referência</em>. Dentro do contexto de gerenciamento de memória, é dito que um objeto faz referência a outro, caso o primeiro tenha acesso a este último (de maneira implícita ou explícita). Por exemplo, um objeto JavaScript tem uma referência ao seu <a href="/en-US/docs/Web/JavaScript/Guide/Inheritance_and_the_prototype_chain">prototype</a> (referência implícita) e para os valores de suas propriedades (referência explícita).</p>
<p>Neste contexto, o conceito de "objeto" se extende para algo mais abrangente do que os objetos comuns do JavaScript, e também contém escopos de função (ou o escopo lexical global).</p>
<h3 id="Referência_de_contagem_do_garbage_collection">Referência de contagem do garbage collection</h3>
<p>Este é o algoritmo mais ingênuo de garbage collection. este algoritmo reduz a definição de "um objeto não mais necessário" para "um objeto não tem outro objeto referenciando ele". Um objeto pode ser coletado pelo garbage collector se não existir referência apontando para este objeto.</p>
<h4 id="Exemplo">Exemplo</h4>
<pre class="brush: js">var o = {
a: {
b:2
}
};
// 2 objetos são criados. Um é referenciado pelo outro como uma de suas propriedades.
// O outro é referenciado pelo fato de ser atribuído à variável 'o'.
// Obviamente, nenhum pode ser coletado pelo garbage collector
var o2 = o; // a varável 'o2' é a segunda coisa que tem
// uma referência ao objeto
o = 1; // agora, o objeto que estava originalmente no 'o'
// tem uma referência única
// encorporada pela variável 'o2'
var oa = o2.a; // Referência para a propriedade 'a' do objeto.
// Este objeto agora tem 2 referências: uma como uma propriedade,
// a outra como a variável 'oa'
o2 = "yo"; // O objeto que estava originalmente em 'o' agora não tem
// nenhuma referência para ele.
// Ele poderia ser coletado pelo garbage collector.
// Entretanto o que era sua propriedade 'a' continua sendo referenciada
// pela variável 'oa', então ele não pode ser coletado.
oa = null; // O que era a propriedade 'a' do objeto original em 'o'
// não tem mais refêrencia para ele. Então pode ser coletado.
</pre>
<h4 id="Limitação_ciclos">Limitação : ciclos</h4>
<p>Esse algoritmo ingênuo tem a limitação de que objetos que referenciam um ao outro (e formam um ciclo), podem ser "não mais necessários" e ainda assim não serem coletados.</p>
<pre class="brush: js">function f(){
var o = {};
var o2 = {};
o.a = o2; // o referencia o2
o2.a = o; // o2 referencia o
return "azerty";
}
f();
// Dois objetos são criados e referenciam um ao outro criando assim um ciclo.
// Eles não vão sair do escopo da função depois dela ser chamada, então eles
// são efetivamente inúteis e podem ser liberados.
// Entretanto, o algoritmo contador de referências considera que desde que ambos
// os objetos sejam referenciados pelo menos uma vez,
// nenhum deles podem ser coletados.
</pre>
<h4 id="Exemplo_da_vida_real">Exemplo da vida real</h4>
<p>Internet Explorer 6 e 7 são conhecidos por terem um coletor com contador de referências para os objetos do DOM. Ciclos são um erro comum que podem gerar erros na memória:</p>
<pre class="brush: js">var div;
window.onload = function(){
div = document.getElementById("minhaDiv");
div<span style="font-size: 1rem;">.referenciaCircular = div;</span>
div.muitosDados = new Array(10000).join("*");
};
</pre>
<p>No exemplo acima, o elemento do DOM "minhaDiv" tem uma referência circular para ela mesma na propriedade "referenciaCircular". Se a propriedade não for removida ou anulada explicitamente, o contador de referências do coletor sempre terá pelo menos uma referência intacta e irá manter o elemento do DOM na memória mesmo se ele for removido da árvore do DOM. Se o elemento do DOM retém muitos dados (ilustrado no exemplo acima com a propriedade "muitosDados"), a memória consumida por esses dados não será liberada.</p>
<h3 id="Algoritmo_de_varredura_e_rotulação">Algoritmo de varredura e rotulação</h3>
<p>Esse algoritmo reduz a definição de "um objeto não é mais necessário" para "um objeto é inacessível".</p>
<p>Esse algoritmo assume o conhecimento de uma lista de objetos chamada <em>roots </em>(raízes) (no JavaScript, o root é o objeto global). Periodicamente, o coletor iniciará por esses roots, encontrando todos os objetos que são referenciados por esses roots, então todos os objetos referenciados por eles, etc. Começando pelos roots, o coletor encontrará todos os objetos acessíveis e coletará todos os objetos inacessíveis.</p>
<p>Esse algoritmo é melhor que o anterior pois "um objeto que não tem referência" leva a esse objeto inacessível. O oposto não é verdadeiro como nós vimos com os ciclos.</p>
<p>A partir de 2012, todos os navegadores modernos vem com um coletor com varredura e rotulação. Todas as melhorias feitas nessa área do coletor do JavaScript (geracional/incremental/concorrência/coletor paralelo) nos últimos anos são implementações melhoradas deste algoritmo, mas não melhorias para o coletor propriamente nem a redução da definição de quando "um objeto não é mais necessário".</p>
<h4 id="Ciclos_não_são_mais_um_problema">Ciclos não são mais um problema</h4>
<p>No primeiro exemplo, depois do retorno da chamada da função, os 2 objetos não são mais referenciados por algo acessível pelo objeto global. Consequentemente, eles não serão acessíveis pelo coletor.</p>
<p>A mesma coisa acontece com o segundo exemplo. Uma vez que o div e o seu gerenciador de eventos se tornem inacessíveis pelos roots, ambos podem ser coletados, apesar de referenciar um ao outro.</p>
<h4 id="Limitação_objetos_devem_ser_feitos_explicitamente_inacessíveis">Limitação: objetos devem ser feitos explicitamente inacessíveis</h4>
<p>Apesar disso ser marcado como uma limitação, isso é algo raro na prática e é por isso que ninguém se importa muito sobre o coletor.</p>
<h2 id="Veja_também">Veja também</h2>
<ul>
<li><a class="external" href="http://www.ibm.com/developerworks/web/library/wa-memleak/">IBM article on "Memory leak patterns in JavaScript" (2007)</a></li>
<li><a class="external" href="http://msdn.microsoft.com/en-us/magazine/ff728624.aspx">Kangax article on how to register event handler and avoid memory leaks (2010)</a></li>
<li><a href="https://developer.mozilla.org/en-US/docs/Mozilla/Performance" title="https://developer.mozilla.org/en-US/docs/Mozilla/Performance">Performance</a></li>
</ul>
|