책을 보다가 abstract, override, virtual이 막막 등장해서 기 기회에 정리해본다


- 부모 함수를 override하려면 부모 함수가 virtual로 정의되어 있어야 한다

- virtual이 아닌 부모 함수를 override하려면 new를 사용해서 override해야 한다

- 자식클래스는 부모 클래스의 virtual 함수를 굳이 override할 필요는 없다. 



- abstract 함수는 반드시 자식 클래스에서 override되야 한다. 안그러면 컴파일러의 질알을 볼 수 있다

- abstract 함수가 포함된 클래스는 abstract 클래스가 되어야 한다.  안그러면 컴파일러의 질알을 또 볼 수 있다



- abstract 클래스에는 함수의 구현이 가능하지만, interface에는 함수의 구현이 불가능하다
Posted by JinnyDown
,
http://kaistizen.net/EE/index.php/imaso/200607_clrprofiler.html

2년 전쯤 필자는 Microsoft SQL Server 2000에 부쩍 관심을 갖고 있었다. 그래서 책을 사서 읽고 데이터베이스 전문 잡지를 구독했는데, 그때마다 저자들의 내공에 깜짝깜짝 놀라고는 했다. 오픈 소스 데이터베이스도 아닌데 내부 구현에 관해 언급하거나, 문서화 되어 있지 않은 기능을 알려주거나 했기 때문이다. 필자는 줄곧 그 점에 의문을 품고 있었다.

그러던 어느 날 SQL Server 전문가로 유명한 정원혁씨의 성능 튜닝 세미나에 참석하게 됐다. 강의 내용도 재미있었지만 필자는 강사의 시범에서 더 놀라운 발견을 할 수 있었다. SQL Server의 내부 이벤트를 잡아낼 수 있는 프로필러를 열어놓고, 관리 도구인 엔터프라이즈 매니저에서 이런저런 작업을 했던 것이다. 예전에는 아이콘을 클릭하면 데이터베이스의 테이블 목록이 나온다는 것을 당연하게 받아들였었다. 그런데 프로필러에는 아이콘 속에 감춰진 쿼리문이 드러나 있었다. 그 순간 필자는 전문가들의 노하우 중 하나를 들여다 봤다는 사실을 깨달았다.

최재훈 | 지난 4월부터 마소의 프로그래밍 노트 연재를 하고 있다. 혹시라도 독자에게 잘못된 내용을 전달하지 않을까 노심초사하며 원고 작업에 몰두해왔다. 덕분에 하루가 멀다 하고 기술문서를 들여다 보고 있다. 독자에게 지식을 전달하는 입장이지만, 필자 자신이 학생인 것 같은 요즘이다.


문제의 해결책을 찾는 것은 쉽다. 구글에서 열심히 검색하면 웬만한 문제는 해결된다. 누군가 똑 같은 경험을 이미 해봤기 때문이다. 그러나 한 시간을 넘게 검색해도 문제의 실마리조차 보이지 않는 경우가 있다. 특히 느려터진 어플리케이션을 손 보는 일만큼 골치 아픈 일도 없다. 문자열의 + 연산이 성능을 저하시킬 수 있다는 사실을 안다고 해도, 그것이 현재 상황을 일으킨 주범이 아닐 수도 있다. 그것 외에도 성능을 저하시킬 수 있는 요인은 얼마든지 있다. 그래서 추측을 하기 시작한다. "IOCP로 구현하면 더 빠르다고 하더라”라는 말에 프로그램을 뜯어고친다. 몇 주간 고생해서 IOCP를 구현했다. 그러나 테스트를 해 보니 그다지 나아진 점이 없다. 몇 번의 시행착오를 더 거친 후에야 데이터베이스 테이블에 인덱스가 잘못 잡혀 있었음이 드러난다.

필자는 지난 2차례의 기사에서 문자열 조작과 관련된 성능 문제를 다뤘다. 그러나 이런 단편적인 지식만으로는 독자가 스스로 미지의 문제를 해결해 나가기 힘들 수도 있다는 생각이 들었다. 그래서 이번 시간에는 사막 한가운데에서부터 방향을 잡고 빠져나올 수 있는 방법을 제시하려 한다.


다양한 성능 분석 도구를 살펴보자.

SQL Server에 프로필러가 있듯이, .NET Framework에도 어플리케이션의 성능을 분석할 수 있는 프로필러가 있다. 필자는 다음과 같은 프로필러를 주로 사용한다. 모두 별도의 비용 없이 마음껏 사용할 수 있는 유용한 도구이다.

  • CLR Profiler

    .NET Framework 1.1과 2.0을 모두 지원한다. 윈도우 서비스나 ASP.NET 웹 어플리케이션도 분석할 수 있다는 것이 가장 큰 장점이다. 이번 기사에서 집중적으로 다루게 될 도구이다.

  • DevPartner Studio .NET Profiler Community Edition

    메모리 뿐만 아니라 실행시간도 한번에 분석한다. Visual Studio .NET 2002/2003과 통합된 환경이 장점이다. 다만 .NET Framework 1.1만 지원한다는 점이 아쉽다.

프로필러가 유일한 성능 분석 도구는 아니다. 역 공학이 적용된 Disassembler로 System.String의 내부 구현을 살펴본다던가, 프로필러로 성능 분석을 할 수 없는 상용 서버의 환경을 성능 카운터로 관찰할 수도 있다. 또한 .NET Framework Class Library (FCL)에 내장된 기능을 사용해서 직접 분석 도구를 작성할 수도 있다.


XML 직렬화 예제 (ConsoleApplication1.exe)

지난 시간에는 서로 다른 XML 직렬화 코드의 성능을 비교해 봤다. 당시 필자는 각각의 경우에 대해 프로파일링 결과를 제시했다. 하지만 기사의 목적이 문자열 처리와 관련된 성능 이슈를 살펴보는 것이었기 때문에 그러한 결과를 얻어낸 과정에 대해서는 언급하지 않았다. 이번에는 최악의 성능을 보여주었던 첫 번째 경우를 프로파일링해 볼 것이다.

지난 기사를 읽지 못한 독자를 위해 다시 한번 <리스트 2>에 일부 소스코드를 적었다. Customer::BuildXml() 메써드를 실행시키면 Customer 객체의 멤버 변수 값을 XML 문자열로 출력한다. 이때 문자열 생성을 위해 System.Text.StringBuilder를 사용하지 않고 단순하게 + 연산(String::Concat)을 했다. 앞으로 기사에 등장하게 될 ConsoleApplication1.exe는 Customer::BuildXml() 메써드를 매개변수로 넘겨 받은 회수만큼 반복 호출하는 콘솔 어플리케이션이다.


<리스트 2> XML 직렬화 예제

public class Customer{	public string Name;	public Address[] Address; 	public string BuildXml()	{		string xmlStr = string.Empty; 		xmlStr += "<?xml version=\"1.0\" encoding=\"utf-16\"?>" + Environment.NewLine;		xmlStr += "<Customer xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns=\"http://example\">" + Environment.NewLine;		xmlStr += "  <Name>" + Name + "</Name>" + Environment.NewLine; 		foreach(Address address in Address) 		{			xmlStr += address.BuildXml() + Environment.NewLine;;		} 		xmlStr += "</Customer>" + Environment.NewLine;;		return xmlStr;	}} ////////////////////////////////////////////////////////////////////////////////////class Program{	[STAThread]	static void Main(string[] args)	{		if(args == null || args.Length < 1) 		{ 			Console.WriteLine("iteration needed!");			return;		}				int iterations = int.Parse(args[0]); 		Customer customer1 = CreateCustomerInstance(); 		for(int i=0; i<iterations; i++)		{			string xmlStr = customer1.BuildXml();		}	}}	

CLR Profiler를 설치하자.

우선 CLR Profiler를 다운로드 받자. .NET Framework 1.1을 위한 프로필러는 CLR Profiler (v1.1)에서, 그리고 2.0을 지원하는 프로필러는 CLR Profiler for the .NET Framework 2.0에서 구할 수 있다. 사용 상의 큰 차이점은 없으므로 여기서는 2.0 CLR Profiler를 기준으로 진행하려 한다. 혹시나 하는 마음에 미리 말해두자면, 1.1 환경에서 컴파일한 어플리케이션을 2.0 프로필러로 프로파일링하려고 시도하면 오류가 발생한다.

물론 반대의 경우도 마찬가지다. 문제는 "이 어플리케이션은 .NET Framework 1.1 환경에서 빌드 되었습니다.”와 같이 친절한 설명을 보여주지 않는다는 점이다. 필자가 테스트를 위해 1.1 환경에서 빌드한 간단한 콘솔 어플리케이션을 2.0 CLR Profiler로 프로파일링했을 때 다음과 같은 오류 메시지가 발생했다. 어디에도 CLR (Common Language Runtime) 버전에 관한 이야기는 없다. 만약 프로필러가 작동하지 않고, 원인을 전혀 짐작할 수 없다면 CLR Profiler의 버전과 프로파일링할 어셈블리 파일의 CLR 버전이 일치하는지 여부부터 점검해봐야 한다.

<리스트 1> 오류 메시지

이 대화 상자 대신 JIT(Just-in-time) 디버깅을 호출하는방법에 대한 자세한 내용은 이 메시지의 뒷부분을 참조하십시오. ************** 예외 텍스트 **************System.NullReferenceException: 개체 참조가 개체의 인스턴스로 설정되지 않았습니다.위치: CLRProfiler.MainForm.WaitForProcessToConnect(String tempDir, String text)위치: CLRProfiler.MainForm.startApplicationButton_Click(Object sender, EventArgs e) 하략……. 

CLR Profiler 설치 폴더에 가면 CLRProfiler.doc 문서가 있다. 영문 문서를 읽는데 애로 사항이 없다면 한번쯤 읽어보기 바란다. 필자가 미처 설명해 주지 못한 부분을 익힐 수 있을 것이다. 100여 페이지에 달하는 분량이 부담스러울 수 있지만, 투자한 것 이상을 돌려 받을 수 있다.


CLR Profiler를 돌려보자.


CLR Profiler
<화면 1> CLR Profiler


Binaries 폴더 안의 CLRProfiler.exe를 실행시켜 보자. 솔직히 CLR Profiler의 외형(<화면 1>)은 전혀 믿음직스럽지 못하다. VS.NET의 유려한 유저 인터페이스와 비교하면 만들다가 만 것 같을 정도다. 하지만 겉모습과는 달리 내실은 튼튼하다. <리스트 2>는 지난 기사에서 제시한 예제 중 일부이다. + 연산, 즉 String::Concat 메써드로 객체의 XML 직렬화를 구현했다. ConsoleApplication1.exe는 Customer::BuildXml을 지정된 회수만큼 반복 실행한다. 이 어플리케이션은 문자열의 생성 실행 회수를 다음과 같이 명령창에서 매개변수로 입력 받는다.

ConsoleApplication1.exe 100000

File/Set Parameters 메뉴를 열어보자. Enter Command Line 텍스트 박스에 매개변수 100000을 입력한다. Set Parameters 메뉴를 자세히 보면 매개변수 외에도 작업 디렉토리(Working Directory)도 설정할 수 있다.

]

이제 File/Profiler Application 메뉴를 열어서 프로파일링할 실행 파일, 이 경우에는 ConsoleApplication1.exe를 선택한다. 파일 탐색기에서 파일을 선택함과 동시에 실행 및 분석 작업이 시작된다. 나중에 좀더 알아보겠지만, File/Profiler Application은 콘솔 및 윈도우 폼 어플리케이션을, Profile Service는 윈도우 서비스(관리도구/서비스 메뉴에서 볼 수 있다.)를, 그리고 Profile ASP.NET은 웹 어플리케이션을 각각 프로파일링한다.

어플리케이션의 실행과 프로파일링 작업이 종료되면 <화면 2>과 같이 결과가 표시된다. 결과가 출력되기까지의 시간이 매우 길게 느껴질 수도 있다. CLR Profiler는 어플리케이션의 성능에 막대한 영향을 미치기 때문에 원래보다 수십 내지는 수백 배 이상 실행시간이 길어질 수 있다. 필자는 ASP.NET 어플리케이션이 수십만 건의 요청을 처리하는 경우를 프로파일링해 본 적이 있다. 저녁 식사를 마치고, 케이블 TV에서 인기 있는 범죄 수사물을 한편 보고 돌아왔을 때 여전히 멈추지 않고 있는 프로필러를 보고 좌절한 적이 있다.

메모.

성능분석 도구가 어플리케이션의 실행 속도를 수십 배나 늦춘다는 것은 언뜻 이해가 안 될 수 있다. 프로필러는 크게 메모리/CPU 분석용으로 나뉜다. 개발하다 보면 소스 코드 중간간에 FCL의 System.DateTime 객체나 C의 time 함수를 넣어서 구간별 실행시간을 재는 경우가 많다. CPU 프로필러는 그와 똑같은 방식으로 작동하고, 앞서 소개한 DevPartner Studio .NET Profiler가 이 분류에 속한다.

CLR Profiler는 이와는 달리 철저하게 CLR 환경의 메모리 운용을 모니터링하는 것에 집중한다. CPU 프로필러처럼 구간별 실행시간을 비교하는 대신, 어떤 객체가 가장 시스템에 부하를 많이 주는지, 그리고 그 객체가 생성되는 구간은 어디인지를 알아낸다. 시스템/어플리케이션 성능은 메모리 및 자원 관리에 영향을 많이 받는다. 그러므로 이렇게 분석된 부분을 집중적으로 개선하면, 적은 노력으로 최대의 효과를 볼 수 있다.


Summary
<화면 2> Summary


관리되는 힙(Managed Heap)과 가비지 콜렉터(Garbage Collector)에 대해 알아보자.

CLR Profiler는 CLR의 메모리 운용 상태를 모니터링하고, 호출 트리, 메모리 할당 등의 상태를 그래프로 화려하게 보여준다. 그래서 별 것도 아닌 듯이 보일 수도 있지만, 프로파일링 결과를 정확하게 분석하기 위해서는 CLR, 그 중에서도 가비지 콜렉터 (Garbage Collector)의 메모리 할당 및 수거 방식에 대해 상세히 알고 있어야 한다. 그러므로 ConsoleApplication1.exe의 프로파일링 결과를 분석하기에 앞서 가비지 콜렉터에 대해 간략히 알아보도록 하자.


가비지 콜렉터의 상태 변화
<리스트 4> 가비지 콜렉터의 상태 변화


<리스트 4>는 CLR의 메모리 관리 방식을 보여준다. CLR은 어플리케이션을 위해 연속적인 메모리 공간을 미리 확보해 놓는다. 그리고 새로운 객체가 생성될 때마다, 즉 new 키워드를 사용할 때마다 뒤쪽의 비어있는 메모리 영역을 부여한다. 동적 할당을 할 때마다 객체 생성에 충분한 메모리 영역을 찾기 위해 링크드 리스트를 뒤져야 하는 C/C++ 환경과는 상당히 다르다.

가용 메모리 영역이 남아 있는 한 별다른 작업 없이 객체가 계속 생성된다. C/C++에서는 사용자가 delete 키워드로 힙 공간의 반환을 명시하지만, 관리되는 환경에서는 사용자가 특정 객체의 메모리 반환을 지시할 수 없다. 그러다가 어느 시점에 이르러 새로운 객체 생성에 필요한 메모리 영역이 부족하게 된다. (<리스트 4> - ①) 이제 가비지 콜렉터가 나설 차례다. 가비지 콜렉터는 우선 필요 없어진 객체와 아직 유효한 객체를 구분한다. ((<리스트 4> - ②) 그리고 유효한 객체만 memcpy로 메모리 영역 앞쪽으로 복사해 놓는다. (<리스트 4> - ③)

<리스트 4>는 상황을 매우 간략하게 묘사하고 있다. 실제로는 한 줄짜리 어플리케이션을 실행시키는 데도 수십 개의 객체가 생성된다. 현실에서는 상호 참조, 소켓 등의 관리되지 않는 자원 해제 등의 문제가 얽혀 있다. 그래서 가비지 수집에 소요되는 시간과 자원이 만만치 않게 된다. 그러므로 성능 향상을 위한 최선의 방법은 가비지 수집의 회수를 줄이고, 자원 해제(Finalization)의 필요성을 줄이는 것이다.

부족한 감이 있지만 일단 이야기 진행에 필요한 만큼 CLR의 메모리 운용 방식에 대해 알아봤다. 앞으로 기회가 있을 때마다 못 다한 설명을 마저 할 생각이다. 하지만 이 기사의 주제는 어디까지나 성능 분석이지 가비지 콜렉터가 아니다. 그러므로 간략하게 설명하고 넘어가거나 아예 언급조차 않는 사항이 있을 수 있다. 만약 가비지 콜렉터에 대해 자세히 알고 싶다면 참고 문헌을 꼭 읽어보기 바란다.


CLR Profiler의 결과를 분석해보자.

<리스트 3> 프로파일링 요약

Allocated bytes:                867,019,216Relocated bytes:                  7,569,048Final Heap bytes:                   244,400Objects finalized:                        0Critical objects finalized:               0Gen 0 collections:                    3,305Gen 1 collections:                        0Gen 2 collections:                        0Induced collections:                      0Gen 0 Heap bytes:                   264,987Gen 1 Heap bytes:                    17,976Gen 2 Heap bytes:                        12Large Object Heap bytes:              8,784Handles created:                         27Handles destroyed:                        0Handles surviving:                       27Heap Dumps:                               0Comments:                                 0

<리스트 3>는 ConsoleApplication1.exe의 프로파일링 결과이다. 이 요약결과를 보면ConsoleApplication1.exe의 생명주기 동안 총 867,019,216바이트의 메모리가 할당됐다. 그이 프로그램이 실행되려면 로컬 컴퓨터의 메모리가 867 메가바이트나 있어야 한다는 뜻은 아니다. 생명주기 동안 몇 번(여기서는 3,305회)이나 가비지 콜렉터가 필요 없어진 객체를 수거해 가기 때문이다.

테스트가 진행된 필자의 컴퓨터에서는 사용 가능한 메모리가 약 400메가바이트 정도였다. 그러나 멀티 태스킹 환경이니만큼 한 어플리케이션 도메인이 모든 가용 메모리를 차지해서는 안 된다. CLR이 ConsoleApplication1.exe 어플리케이션 도메인을 위해 배당한 메모리는 Gen n Heap bytes와 Large Object Heap bytes의 합으로 생각할 수 있다. 각각의 수치는 어플리케이션 생명주기 동안의 평균값이다. 즉 264,987 + 17,976 + 12 + 8,784 = 291,759 바이트의 메모리로 867,019,216 바이트의 객체를 생성하고 파괴한 것이다. 단순한 계산으로는 최소한 867,019,216 / 291,759 = (약) 2970회의 가비지 수집이 일어났을 것임을 알 수 있다. 실제로 Gen n collections 값은 3,305회나 가비지 수집이 이뤄졌다는 것을 알려준다.


Objects by Address
<화면 3> Objects by Address


View/Objects by Address 메뉴(<화면 3>)를 선택하면 특정 순간, 기본적으로는 프로그램 종료 시점의 관리되는 힙의 상태를 보여준다. 앞서 살펴본 <리스트 4>와 비슷하다. 왼쪽 그래프가 일반적인 객체 생성을 위해 사용되는 관리되는 힙이다. 그래프 오른쪽에 조그맣게 gen 0, gen 1이라고 표시되어 있다. 일반적인 어플리케이션은 gen 2 영역도 갖기 마련이지만 ConsoleApplication1.exe는 문자열만 생성하는 간단한 테스트 프로그램이라 gen 2 영역이 생성될 기회가 없었다.

ConsoleApplication1.exe의 프로파일링 요약 정보에서 가장 주목해야 할 부분은 Allocated bytes, Relocated bytes, 그리고 Gen 0 collections이다. 값이 0이 아닌 항목 중에서는 이 세 개가 중요한 의미를 담고 있다. Allocated bytes와 Gen n collections는 앞서 살펴봤다. Relocated bytes는 가비지 콜렉터가 memcpy로 위치를 옮긴 객체의 크기이다. 가비지 수집에서 살아남은 객체는 메모리 영역 앞쪽으로 복사됨을 보았다. 몇몇 예외적인 경우에는 살아남은 객체가 복사되지 않고, 원래 자리에 남아있기도 한다. 그러므로 Relocated bytes를 가비지 수집에서 살아 남아 다음 세대로 격상된 객체의 크기라고 보아도 무방하다.

요약 정보를 종합해 보면 ConsoleApplication1.exe는 메모리 소모적 어플리케이션이다. Customer::BuildXml() 메써드를 10000번 호출하는데 무려 867 메가바이트나 필요하다. 그래서 관리되는 힙(xxx Heap bytes)이 금방 꽉 차게 된다. 이에 따라 가비지 수집이 자주 발생하게 되고 성능이 저하되는 것이다. 시간당 트랜잭션이 100만 건인 XML 웹 서비스에서 이런 일이 발생하면, ConsoleApplication1.exe와 상용 웹 서비스 어플리케이션의 복잡도를 감안하면 서비스 자체가 불가능할 수 있다는 결론이 나온다.

다만 Relocated bytes/Allocated bytes의 값(8%)이 작다는 사실은 양호하다고 봐야 한다. 생성된 객체 중 극히 일부분만 다음 세대까지 살아남는다면, 그만큼 memcpy로 복사하는 노력이 적게 들기 때문이다. ConsoleApplication1.exe에서 생성되는 XML 문자열은 for 문 영역 안에서만 참조가 유효하기 때문에 이런 결과가 나온 것이다.


Time Line
<화면 4> Time Line


<리스트 5> Time Line 그래프 읽기

수평축에 작은 글씨로 gc #1, gc #2라고 적혀 있다. 이것은 가비지 수집이 일어난 시점을 알리고 있고, 그래프의 높이가 급격히 줄어드는 시점과 정확히 일치한다. #n은 어플리케이션의 생명주기 내에서 몇 번째 발생한 가비지 수집인지를 알려준다. 같은 시간 내에서라면 #n 값이 적을수록 성능 향상에 도움이 된다는 것은 강조할 필요조차 없다.

<화면 4>에서는 모든 gc #n 글자가 붉은 색이다. 붉은 색 글자는 1세대 영역에 대한 가비지 수집 작업을 뜻한다. 2세대 영역과 3세대 영역에 대한 각각 녹색과 파란색으로 표시된다. 다음 기회에 자세히 알아보겠지만 n 세대 영역에 대한 가비지 수집은 k ( k <= n ) 세대 모두에 대해 메모리 수거를 하기 때문에, 2세대는 1세대보다 그리고 1세대는 0세대의 경우보다 느리다.


<화면 4>의 Time Line 그래프(메뉴: View/Time Line)는 앞서의 분석을 한눈에 알기 쉽게 보여준다. 최초 구간(0.0 sec ~ 0.4 sec)에서는 메모리 요구량이 매우 적다. 어플리케이션의 초기화 구간이다. 다음 구간부터는 그래프가 송곳처럼 날카로운 경사를 보인다. 0.45 sec ~ 0.5 sec 사이에 약 4차례의 가비지 수집이 이뤄졌다. 전체 실행시간 중에 가비지 수집이 차지하는 비율이 매우 높아서 어플리케이션의 본래 임무에 투자되어야 할 귀중한 자원이 낭비되고 있는 상황이다.

이상적인 어플리케이션이라면 Time Line 그래프는 <화면 5>와 같을 것이다. <화면 5>는 완만한 메모리 사용 증가율을 보인다. 전체 실행시간 중 가비지 수행이 차지하는 비중은 매우 작다. 0.1 sec에 10여 차례나 가비지 수집을 했던 <화면 4>에 비해, <화면 5>는 1.5 sec에 한 차례 정도밖에 가비지 수집을 하지 않는다. <화면 5>에서 한 차례 가비지 수집을 하는 동안, <화면 4>에서는 무려 150차례나 하는 꼴이다.


이상적인 Time Line 그래프
<화면 5> 이상적인 Time Line 그래프


글을 접으며

이번 기사에서는 수많은 성능 분석 도구 중에서 CLR Profiler를 선택하여 기본적인 사용법에 대해 알아봤다. 또한 가비지 콜렉터의 동작 방식도 살짝 들여다 봤다. 프로필러를 처음 접하는 개발자에겐 약간 어려운 내용이었을지도 모르고, 성능분석을 여러 차례 해본 중급 개발자에겐 너무 쉬운 이야기였을 것이다. 만약 독자가 CLR Profiler를 처음 접한다면 반드시 기사의 내용을 실습해보고, 기사나 참고 문헌을 한번씩 읽고 숙지하기를 간절히 바란다. 다음 기사에서는 성능분석 문제를 보다 깊이 파헤칠 것이기 때문이다.


남기는 말

기사 내용 중 이해가 안 되는 내용이 있다면 필자의 이메일이나 블로그로 문의해주기 바란다. 필자의 취미는 질문에 대답하는 것이다.


참고 문헌


피드백과 지원

이 문서에 대한 피드백이나 지원을 하고 싶으신 분은 저자에게 연락해주시기 바랍니다. 발전적인 정보 제공 및 건의는 언제나 환영합니다.

Posted by JinnyDown
,
[C#.NET 고급문법]
  [HOONS] 가비지 컬렉터의 동작원리
 작성자 : 박경훈(HOONS)  작성일 : 2007-08-30 오후 5:53:17
 E-mail : hoonsbara앳hotmail.com  Homepage : http://www.hoons.kr


흔히 닷넷기반의 환경을 관리되는(Managed) 환경으로 부르곤 한다. 그렇다면 도대체 무엇이 관리가 된다는 것인가? 그 관리의 주체는 바로 메모리이다. 이전 Native 시대에는 메모리를 할당하고 해제하는 부분을 개발자가 직접 처리하였지만 닷넷은 그 부분을 자동으로 관리해주는 것이다. 닷넷이 관리되는 환경의 수행이 가능할 수 있는 것은 바로 “가비지 컬렉터(Garbage Collector)”가 닷넷에 존재하기 때문이다. 관리환경의 장점을 최대한 활용하고 사용하기 위해서는 가비지 컬렉터에 대해서 잘 알아 두어야 하고 동작원리를 파악하고 있어야 한다. 필자는 가비지 컬렉터의 원리에 대한 내용을 2002년도 “Chappell의 .NET 여행”이라는 책에서 처음 접했었고, 이 내용은 닷넷의 메모리 관리를 이해하는데 어느 정도의 기반지식이 될 수 있었다. 

1. 가비지 컬렉터와 가비지 컬렉션

가비지 컬렉터는 앞에서 설명한 것과 같이 메모리를 관리해주는 메카니즘이다. 많이 혼동하는 부분이 바로 가비지 컬렉터와 가비지 컬렉션일 수 있다. 가비티 컬렉터는 Mark&Compact 알고리즘을 이용하여 객체들의 관계를 추적한다. 즉, 인스턴스화 시켰던 DataSet에 null을 할당하면 DataSet은 사용하지 않는 객체로 간주되고 가비지 컬렉션시에 메모리 해제의 대상이 된다. 하지만 DataSet은 DataTable을 가지고 있고, DataRow, DataItem과 같은 여러 객체들을 참조하고 있다. 바로 DataSet이 해제가 되면 그와 상호관련이 있었던 모든 객체들 역시 메모리를 해제해야 할 것이고 바로 가비지 컬렉터는 이러한 복잡한 관계를 Mark&Compact 알고리즘을 이용해서 이해하고 각각 해제될 수 있는 것이다. 

가비지 컬렉션은 바로 메모리를 해제하고 새롭게 재배치 하는 작업을 칭하고 자동으로 수행되지만 수동으로도 수행을 명령할 수도 있다. 가비지 컬렉션이 수행되는 시기는 개발자가 알 수 없지만 분명한 것은 메모리가 부족하면 분명 가비지 컬렉션이 일어난다는 것이다. 

System.GC.Collect();

가비지 컬렉터는 세대별로 나누어서 메모리를 관리한다. 즉, 메모리를 관리하는 그릇이 3개가 존재한다고 보면 된다. 세대는 0,1,2 세대로 나누어지고 최초의 메모리는 무조건 0이라는 공간에서 관리가 된다고 보면 된다. 그리고 가비지를 한번 정리하였지만 해제되지 않은 객체는 바로 다음 세대로 이동하게 된다. 가비지 컬렉션은 세대별로 독립적이라고 보면 된다. 즉, 가비지 컬렉션이 일어날 때 0,1,2 세대 별로 동시에 발생하는 것이 아니라 개별적으로 가비지 컬렉션을 수행한다는 것이다. 최근의 생긴 메모리 수록 즉, 0세대일 수록 컬렉션이 많이 발생한다. 이것은 당연한 동작 원리이다. 보통 한번 사용한 객체는 꾸준히 많이 사용하지만 그렇지 않은 객체는 단기적으로 사용을 많이 하기 때문이다. 다음 [그림1]은 가비지 컬렉터가 가비지 컬렉션을 수행하는 장면을 보여주고 있다.


[그림1]가비지 컬렉션의 동작

이렇게 수행을 하고 남은 객체는 바로 그 위의 세대로 승격된다. 다음 [그림2]를 살펴보자.




[그림2]다음 세대로 승격


이 승격은 2세대까지 2번 승격된다. 0세대는 활발하게 가비지 컬렉션이 수행되지만 2세대는 거의 발생되지 않는다고 보면 된다.


2. 메모리의 효율적인 사용

앞에서 살펴본 가비지 컬렉터는 관리되는 환경의 모든 메모리를 책임지고 관리한다. 이전 Native 시대에서는 메모리 누수와 같은 문제가 발생하면 개발자의 책임이었지만 닷넷에서는 그 책임을 대신 주어준다는 것이다. 하지만 관리되어지는 환경이라고 해서 모든 것을 전적으로 가비지 컬렉터에 의존해서는 안 된다. 물론 가비지 컬렉터가 하드웨어가 무척 발전한 지금의 환경에 부응하고 있는 기능임은 틀림없다. 하지만 필자가 말하고자 하는 것은 가비지 컬렉션이 언제 어떤 객체를 수집하고 어떻게 동작하는 것임을 알아두고 그에 맞추어 코드를 작성해야 한다는 것이다. 이런 효츌적인 메모리 관리를 위해서 몇 가지 지침들을 적어보도록 하겠다.

- 코드에서 System.GC.Collect()를 이용해서 직접 가비지 컬렉션의 수행을 호출하는 것은 가급적 피한다. 

닷넷에 연고가 있는 독자라면 가비지 컬렉션을 수동으로 수행하면 안좋다 라는 말을 많이 들어봤을 것이다. 그럼 이제 그 이유를 살펴보자. 이유는 크게 두 가지 이유가 있다. 첫 번째는 가비지 컬렉션이 수행하는 메카니즘이 생각보다 간단하지 않기 때문이므로 성능상 부하가 있을 수 있다. 왜냐하면 현재 객체가 사용 중인지 확인하는 작업이 필요하며 그 작업이 끝난 후에 객체를 파괴한다. 객체를 파괴할 때에는 참조되고 있는 객체 역시 파괴해야 한다는 것이다. 뿐만 아니라 파괴된 객체들의 빈자리를 매꾸기 위해서 가비지 컬렉터는 객체들을 재배치 작업(Compaction 작업)을 수행하게 되므로 메모리가 많을수록 그 부하가 클 수 있기 때문이다. 두 번째 이유는 객체가 승격되기 때문이다. 객체가 0세대에서 한번 승격되면 그 객체가 더 이상 사용하지 않는다 하더라도 자동적으로 그 객체가 정리될 확률이 줄어들기 때문이다. 그렇기 때문에 코드에서 직접 가비지 컬렉션의 수행의 호출을 추천하지 않는 것이다. 

- 사용하지 않는 객체는 가비지 컬렉터가 수집할 수 있는 대상으로 설정한다.

그렇다면 가비지 컬렉터가 수집하는 대상은 어떤 대상인가? 좀 더 쉽게 이해하기 위해서 다음과 같은 클래스를 만들었다고 가정하자.


Class A
{
  string a="HOONS"
}
Main메서드()
{
  A aClass= new A();//클래스 생성
}

그렇다면 여기서 aClass객체를 사용하고 있는지 어떻게 계산할 것인가이다. 가장 쉬운 방법은 aClass에 null을 대입해 주면 되기도 하지만 가장 바람직한 방법은 using을 이용해서 객체를 사용하는 범위를 지정해주는 것이다. using은 IDisposable 인터페이스를 상속받아서 Dispose() 메서드를 구현해야 한다. 그리고 Dispose()에서는 클래스 안에서 사용했던 자원을 해제하는 것이다. finalizer(~생성자())를 구현한다면 가비지 컬렉터는 자원을 해제하기 바로 직전에 수행하게 될 것이다. 이 finalizer에서는 만약 Dispose()를 호출하지 않을 것을 대비해서 Dispose()를 수행하는 용도나 아니면 관리되지 않은 영역 즉, unmanaged 자원이 있다면 이 자원을 호출하면 된다. 


- 전역변수는 가급적 초기화하지 않는다.

다음과 같은 코드가 있다고 가정하자.

Class A
{
  ArrayList a=new ArrayList();
  public A(int Length)
  { 
      a=new ArrayList(Length); 
  }
}


이 코드는 내부적으로 다음과 같이 동작된다. 

Class A
{
  ArrayList a=new ArrayList();
  public A(int Length)
  {
      a=new ArrayList();
      a=new ArrayList(Length); 
  }
}

그렇기 때문에 필요없는 가비지가 생기게 되는 것이다.




- 자주 사용하는 객체는 전역변수로 잡아서 가비지를 최소화한다.

자주 사용하는 객체는 여러 개를 만들어 사용할 없다는 것이다. 이 부분은 이론상 쉽게 이해할 수 있으므로 깊게 언급하지 않겠다.



3. CLR 프로필러

그렇다면 가비지 컬렉터에서 내부적으로 관리되고 있는 메모리를 어떻게 들여다 볼 수 있는 것인가? 이것은 Microsoft에서 제공하고 있는 CLR 프로필러(Profiler)라는 툴을 이용하면 된다. 이툴은 MS 다운로드 사이트에서 다운 받을 수 있으며 다음 [화면1]은 CLR 프로필러를 실행한 화면을 보여주고 있다. 



[화면1] CLR 프로필러

이 프로필러는 각 세대별로 가비지 컬렉션이 일어난 횟수를 보여주고 있고 현재 각 세대별로 차지하고 있는 힙의 크기 그리고 할당되고 재할당 된 메모리 바이트를 보여주고 있다. 뿐만 아니라 참고 그래프를 제공하고 있기 때문에 메모리 사용을 분석하는데 큰 도움을 줄 수 있다. 이 툴의 자세한 내용은 이전 HOONS 닷넷에서 진행한 세미나에서도 언급한 적도 있고, 이전 마소에서도 다루었던 적이 있으므로 자세한 사용방법은 지면상 생략하도록 한다.

Posted by JinnyDown
,
...
Posted by JinnyDown
,
...
Posted by JinnyDown
,
...



공식 한국 싸이트
http://gimp.kr/

메뉴얼
http://docs.gimp.org/
Posted by JinnyDown
,
Posted by JinnyDown
,


'<기타> > ___웃긴거' 카테고리의 다른 글

먹이를 노리는 사자  (0) 2009.07.09
개깜짝  (0) 2009.07.09
요즘 피씨방 알바 공고  (0) 2009.03.23
OO 고추가 더 맵다  (0) 2009.03.23
역사속의 락커들  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

개깜짝  (0) 2009.07.09
호나우도 최악의 날  (0) 2009.03.23
OO 고추가 더 맵다  (0) 2009.03.23
역사속의 락커들  (0) 2009.03.23
비틀즈 내한공연 인증샷  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

호나우도 최악의 날  (0) 2009.03.23
요즘 피씨방 알바 공고  (0) 2009.03.23
역사속의 락커들  (0) 2009.03.23
비틀즈 내한공연 인증샷  (0) 2009.03.23
민망한 동화  (0) 2009.03.23
Posted by JinnyDown
,

 




My name is Maximus Decimus Meridius,
나의 이름은 막시무스 데시무스 메리디우스,

Lead singer of the Armies of the North,
북방군단의 리드 보컬이었고,

Guitar of the Felix Legions,
펠릭스 군단의 기타리스트였으며,

loyal session to the true music label, Marcus Aurelius.
진정한 음악 레이블 사였던 마르쿠스 아우렐리우스의 전속 세션이었다
.
.
.
And I will have my concert,
그리고 반드시 나의 콘서트를 가질 것이다

in this life or the next.
이번 생에서 안된다면 다음 생에서라도






조지 워싱턴의 AC/DC 가족밴드






단테의 신곡(新曲)






Davinci Code A Minor






제국주의자 종간나 새끼들에게
내래 인민의 락을 보여주갔어






이집트 시대 귀족들은 락을 할 줄 아는 게
하나의 당연한 의무이자 특권이었다 





태초에 보컬과 기타리스트가 있었다
만들고 나서 하나님이 보시니 그 모습들이 보기 좋았더라






스탈린 (Joseph Stalin, 1879~1953)
스탈린의 일대기는 멜로디를 중시하는 소비에트식의 악곡으로 인해
오랫동안 사랑 받아왔다
그 '전설'은 영웅적인 우드스탁 혁명의 모의자이자 전설적인 기타리스트인
레닌의 충실한 추종자로서 그의 위대성을 입증받고 있다





내 사전에 스튜디오 오버레코딩 이란 없다





공연중이던 번개의 신 토르는 공연중에 캠코더로 영상을 찍던 찌질이에게
욕설과 폭력을 행사하였다






운명의 세 여신에게 쫓기고 있는 엘비스 프레슬리





모나리자는 그룹 KISS의 열렬한 추종자였다





모세가 시나이 산에서 신에게 받은 기타로 공연을 펼치고 그 기타를 성궤에 보관하였더라
이로서 야훼께서는 성궤 안의 기타를 야훼와 유대 민족 사이의 약속의 증거로 삼으셨도다





바로크 메탈을 연습중이신 루이15세의 애첩 마담 퐁파두르(Madam Pompadour)





이집트 왕립 합창단의 촌철살인급 간지




알거 다 아는 큐피드와 프시케



그리고 대망의 마지막 걸작
.
.
.
.
.


W
hen I find myself in times of trouble
내가 졸라 골치아픈 순간에


"Mother Mary" comes to me
어머니(성모 마리아)가 나에게 오셨지


Speaking words of wisdom
지혜의 말 한 마디를 건네며


Let it be
내비 둬

 

 

Jesus Christ SuperStar

'<기타> > ___웃긴거' 카테고리의 다른 글

요즘 피씨방 알바 공고  (0) 2009.03.23
OO 고추가 더 맵다  (0) 2009.03.23
비틀즈 내한공연 인증샷  (0) 2009.03.23
민망한 동화  (0) 2009.03.23
[스크립] 연애인 짤방  (0) 2009.03.23
Posted by JinnyDown
,


'<기타> > ___웃긴거' 카테고리의 다른 글

OO 고추가 더 맵다  (0) 2009.03.23
역사속의 락커들  (0) 2009.03.23
민망한 동화  (0) 2009.03.23
[스크립] 연애인 짤방  (0) 2009.03.23
문제의 L양 사진  (0) 2009.03.23
Posted by JinnyDown
,
이름만 들어도 다 아는 "B양" 7명의 남자와 동거..........
[백설공주와 일곱난쟁이]
 
 
 
그를 처음 봤을때.. 전 알몸 이었어요....................
[인어공주]
 
 
 
그녀는 왜 12시에 구두를 흘리고 뛰쳐 나왔는가...........
[신데렐라]
 
 
 
"Y양"토끼를 따라 굴속으로 들어가는데...................
[이상한 나라 앨리스]
 
 
 
그녀의 침대에 늑대가?..................................
[빨간모자와 늑대]
 
 
 
열쇠구멍으로 들여다본 그의 집..........................
[아기돼지 삼형제]
 
 
 
그녀는 그의 무엇을보고 그를 사랑하게 되었는가..........
[미녀와 야수]
 
 
 
그의 길어지는 그곳의 비밀..............................
[피노키오]
 
 
 
벌거벗고 길을 방황한 한남자의 이야기...................
[벌거벗은 임금님]
 
 

S양 쌀삼백석에 몸을 팔아?.......
[심청전]
 
 

그들만의 레이스 누가 더 빨리 지칠 것인가...
[토끼와 거북이]
 
 
 
사과를 먹고 쓰러진 그녀를 성으로 데려가는데
[백설공주 2]
 
 
 

약속한 100일이 지나 여자몸으로 바뀌었는지를 확인하러 동굴 속으로 들어가는데...
[단군신화]
 
 
 
새어머니와 새언니에게 고통받는 그녀
[콩쥐팥쥐- SM ver.]
 
 
 
아직 어린 오누이인 그들 숲속에서 왜 헤매고 있을까?
[헨젤과 그레텔]
 
 
 
여탕 훔쳐보기
[선녀와 나무꾼]
 
 

그것은 하늘 높은 줄 모르고 치솟게 되는데 ...
[재크와 콩나물]
 
 

무방비 상태로 누워있는 그녀. 그녀에게 다가가는 검은 그림자.
[잠자는 숲속의 미녀]
 
 
 
작아졌다 커지는 비밀
[호호 아줌마]
 
 
 
C양, B씨와의 부적절한 관계
[춘향전]
 
 

그의 취향은 .... 동물...?
[백조의 호수]
 
 

떡친 호랑이와 두 남매의 늦은밤 만남
[햇님 달님]
 
 

밤마다 긴머리를 늘어뜨리고 남자를 기다리는 그녀 ,,
[라푼젤]
 
 

팬티조차 입지 않게된 사연 ,,
[곰돌이 푸우]
 
 

어느 한나라의 임금님에게는..말도 못하게 큰 무언가가 달려 있다는데...
[임금님귀는 당나귀귀]
 
 

어느 천재 박사가 너무 외로워 인조인간을..만들었는데.....
[프랑켄슈타인]
 
 

이른 아침부터 형수는 주걱을 손에 든체, 시동생과 헐떡이는데...
[흥부와 놀부3]
 
 

A군...무언가를..부드럽게 만지작 거리는데....
[알라딘]
 
 

어둠이 세상을 삼켜버린 늦은 시각.
두 남매는 사또와의 은밀한 만남을 위해 발걸음을 재촉하는데...
[장화홍련]
 
 

동물과 잠자리를 한 남자...
[구미호]
 
 
 
늦은밤...그녀는 시계탑으로 올라가는데...
[노틀담의 꼽추]
 
 

왼손은 거들뿐...
[슬램덩크]
 
 
 
나에게...사이즈가 딱 맞는 느낌,,아 ...
[구두를 찾은 신데렐라]
 
 
 
그의 우렁찬 한마디에 서서히 벌리기 시작했다
[알리바바와 40인의 도둑中 열려라 참께 기도]
 
 
 
H군 자신의 그것을 계속 만지작 거리는데...
[혹부리 영감]
 
 

동굴안 감금.. 100일.. 결국 여자가 되다..
[단군신화] 2
 
 
 
내가 맨날 칠수잇엇던 이유는 매일 칠수 있다는 자신감때문이었는데...
[홈런왕 이승엽]
 
 

그가 속삭일수록 점점 더 조여오는데...
[삼장법사]

'<기타> > ___웃긴거' 카테고리의 다른 글

역사속의 락커들  (0) 2009.03.23
비틀즈 내한공연 인증샷  (0) 2009.03.23
[스크립] 연애인 짤방  (0) 2009.03.23
문제의 L양 사진  (0) 2009.03.23
Y의 배신  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

비틀즈 내한공연 인증샷  (0) 2009.03.23
민망한 동화  (0) 2009.03.23
문제의 L양 사진  (0) 2009.03.23
Y의 배신  (0) 2009.03.23
편의점 알바  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

민망한 동화  (0) 2009.03.23
[스크립] 연애인 짤방  (0) 2009.03.23
Y의 배신  (0) 2009.03.23
편의점 알바  (0) 2009.03.23
좋은 패쓰다  (0) 2009.03.23
Posted by JinnyDown
,

Y의 배신

<기타>/___웃긴거 2009. 3. 23. 16:52


'<기타> > ___웃긴거' 카테고리의 다른 글

[스크립] 연애인 짤방  (0) 2009.03.23
문제의 L양 사진  (0) 2009.03.23
편의점 알바  (0) 2009.03.23
좋은 패쓰다  (0) 2009.03.23
시간을 멈추는 드리블  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

문제의 L양 사진  (0) 2009.03.23
Y의 배신  (0) 2009.03.23
좋은 패쓰다  (0) 2009.03.23
시간을 멈추는 드리블  (0) 2009.03.23
빨랑 다음 페이지로 넘겨  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

Y의 배신  (0) 2009.03.23
편의점 알바  (0) 2009.03.23
시간을 멈추는 드리블  (0) 2009.03.23
빨랑 다음 페이지로 넘겨  (0) 2009.03.23
요가 화이어~~~  (0) 2009.03.23
Posted by JinnyDown
,

'<기타> > ___웃긴거' 카테고리의 다른 글

편의점 알바  (0) 2009.03.23
좋은 패쓰다  (0) 2009.03.23
빨랑 다음 페이지로 넘겨  (0) 2009.03.23
요가 화이어~~~  (0) 2009.03.23
으악!! 악어가 쫓아온다!!  (0) 2009.03.23
Posted by JinnyDown
,


'<기타> > ___웃긴거' 카테고리의 다른 글

좋은 패쓰다  (0) 2009.03.23
시간을 멈추는 드리블  (0) 2009.03.23
요가 화이어~~~  (0) 2009.03.23
으악!! 악어가 쫓아온다!!  (0) 2009.03.23
소프트웨어 유지보수란?  (0) 2009.03.23
Posted by JinnyDown
,