MinUk.Dev
Javascript Memory Leak - minuk dev wiki

Javascript Memory Leak

created : Tue, 07 Apr 2020 20:16:46 +0900
modified : Tue, 15 Mar 2022 02:23:25 +0900

주제 후보

  • 메모리 누수란 왜 일어나는가? - garbage collector의 동작방식, javascript 에서 메모리 누수 없이 코딩하는 습관 가지기
  • 비동기처리하기 - javascript callback, promise, async-await
  • 웹사이트에서 원하는 정보 추출하기 - 웹사이트 크롤링, 파싱하기
  • 나무위키 유의어, 반의어 - Word2Vec를 통한 나무위키 데이터 학습

→ 익명의 대학 동기에게 재밋는 주제를 고르라고 해서 진행되었습니다.

메모리 누수란 왜 일어나는가?

결론부터 말하자면 프로그래머의 실수. 코딩 습관을 잘 들이자.

  • Garbage Collection 의 필요성
    • C언어에서 메모리를 직접 할당하고 해제하는 malloc, free 라는 방식을 사용하였다.
#include <stdio.h>
#include <stdlib.h>

int main(int argc, char** argv) {
    int n;
    scanf("%d",&n);
    int* arr = (int*)malloc(sizeof(int) * n);
    // do something
    free(arr);
    return 0;
}
- → 소프트웨어의 규모가 커지고 협업이 증가하면서 메모리를 책임지지 못하게 되었다.
- →Garbage Collection의 필요성 (cf. rust 의 memory ownership)
    - 단, 최초의 GC는 LISP 이라는 언어에서부터 있었다고 알려져있음.(1958년)

Garbage Collection

초기 아이디어 - 참조 카운트(Reference Counting)

A a = new A();
a.b = new B();
a.b.c = new C();


a.b.c = null;

f();


void f() {
  A a = new A();
 // do something
}

이 때 f 메서드 바깥에서는 a가 사용되지 않는다.

a가 가르키던 메모리 공간에 대해서 f가 끝나면 reference count가 감소하여 reference count가 0이 되었다고 판단하고 gc가 할당을 해제한다.

  • 문제점 - 순환 참조 (Circular Referencing)
A a = new A();
B b = new B();
a.refB = b;
b.refA = a;
a = null;
b = null;

a와 b의 공간에 더이상 접근가능한 객체가 없음에도 불구하고 a와 b의 reference count 는 각각 1이기 때문에 gc가 메모리 해제를 하지 않는다.

  • cf. c++ shared_ptr 이 이러한 방식을 사용한다.

도달 가능한 (reachable) 객체만 남기자 - Mark and Sweep

gc.png

출처 : https://en.wikipedia.org/wiki/Tracing_garbage_collection#Naïve_mark-and-sweep

  • Root Set - 항상 유요한 최초의 참조
  • 만약 RootSet 에서부터 참조 가능한 모든 객체를 Mark 했는데 Mark가 안된 Object가 있다면 그 객체는 도달 불가능한 객체(unreachable)이므로 삭제(Sweep)한다.
  • cf. Copying Algorithm → Fragementation 문제 해결하는 방법 (Compaction)

단점

  • 그러면 gc가 돌때마다 모든 Object 를 전부 찾아봐야한다.

Generational Algorithm

  • Weak generational hypothesis

    • 대부분의 객체는 금방 Garbage가 된다.
    • 오래된 객체들이 최근에 생성된 객체로의 참조는 거의 없다.
  • 젊은 세대 (young generation)과 오래된 세대(old generation)으로 객체를 구분하고 각 세대별로 처리 작업을 다르게 하자

    • 젊은 세대들은 훨씬 더 collection 이 자주 일어난다. (minor collection) - 대부분의 객체들은 금방 Garbage 가 된다.
    • 오래된 세대들은 상대적으로 적게 일어난다 (major collection)
    • 살아남은 젊은 세대들은 오래된 세대가 된다.(물론 오래된 세대가 되는데에 대한 규칙은 다들 약간씩 다를수 있다.)
  • 참고글

데블스캠프2017/Internal Of Java Script Core’s Garbage Collector

자바스크립트에서 메모리 누수의 4가지 형태

NAVER D2

[Rust] 러스트의 꽃, Ownership 파헤치기

Memory Management

https://www.ps.uni-saarland.de/courses/gc-ws01/slides/generational_gc.pdhttps://www.ibm.com/developerworks/web/library/wa-memleak/wa-memleak-pdf.pdf

function foo(arg) {
  bar = "global variable";
}


function foo(arg) {
  window.bar = "global variable";
}

var someResource =getData();
setInterval(function() {
  var node = document.getElementById('Node');
  if (node) {
    // something
    node.innerHTML = JSON.stringify(someResource);
  }
}, 1000);

Garbage collection

mbbill/JSC.js

https://www.ibm.com/developerworks/web/library/wa-memleak/wa-memleak-pdf.pdf