본 내용은 "델파이 프로그래밍 언어" 도서의 일부분을 정리한 것입니다. 보다 자세한 내용은 해당 도서를 통해 확인할 수 있습니다.

 "델파이 프로그래밍 언어"(342쪽, 구입안내): http://tech.devgear.co.kr/devgearbook/2431 


 그 외 데브기어 도서들은 다음 링크를 통해서 확인할 수 있습니다: www.devgear.co.kr/book


이벤트로서의 익명 메소드


메소드 참조를 사용하는 동기는 연결된 변수를 포함할 수 있는 타입을 가지기 위해서입니다. 이것은 클로저(closure)라고도 알려져 있습니다. 클로저는 그 정의된 환경을 정의된 시 점에 참조된 모든 지역 변수들과 함께 가두기 때문에, 해제되어야 할 상태를 가지게 됩니다. 메소드 참조는 매니지드 타입이므로(참조 카운트가 됩니다), 메소드 참조는 이 상태를 추적 하여 필요할 경우 해제할 수 있습니다. 메소드 참조나 클로저가 이벤트 같은 메소드 포인터 에 자유롭게 대입될 수 있으면, 엉뚱한 위치를 가리키는 포인터(dangling pointer)나 메모 리 누수(memory leak)를 가진 잘못된 프로그램을 만들기 쉬워집니다.


델파이의 이벤트는 속성에 어떤 조건이 추가된 것입니다. 타입의 종류 외에는 이벤트와 속성 사이에는 아무런 차이가 없습니다. 어떤 속성이 메소드 포인터 타입이면 이벤트가 됩니다. 속성이 메소드 참조 타입이면, 논리적으로 이벤트도 고려해야 합니다. 하지만 IDE는 메소 드 참조 타입을 이벤트로 다루지 않습니다. 이것은 IDE에 컴포넌트와 커스텀 컨트롤로 설 치된 클래스에 대해 중요합니다.


따라서, 메소드 참조나 클로저 값으로 대입될 수 있는 컴포넌트나 커스텀 컴포넌트에서 이 벤트를 가지려면, 그 속성은 메소드 참조 타입이어야 합니다. 하지만, IDE가 이벤트로 인식 하지 못하기 때문에 이런 방법은 편리하지 않습니다. 


아래의 코드는 메소드 참조 타입의 속성을 사용하여 이벤트로서 동작하는 예제입니다.


anonymous_3_1.png

   

변수 바인딩 메커니즘


메모리 누수(memory leak)가 발생하는 것을 피하기 위해, 변수의 바인딩 과정을 더 자세 히 알아보는 것도 유용합니다.

프로시저, 함수 혹은 메소드(이하“루틴”)의 시작 부분에서 정의된 지역 변수들은 일반적으 로 해당 루틴이 활성인 상태에서만 존재합니다. 익명 메소드는 이런 변수들의 수명을 늘릴 수 있습니다.


익명 메소드가 자신의 바디 내에서 외부의 지역 변수를 참조하면, 그 변수는“캡쳐 (capture)”됩니다. 캡쳐는 변수의 수명을 연장하는 것을 의미하므로, 선언된 루틴과 함께 사라지지 않고 익명 메소드 값만큼의 수명을 가지게 됩니다. 변수 캡쳐는 값이 아니라 변수 를 캡쳐한다는 것에 주의하십시오. 익명 메소드를 구축하여 변수가 캡쳐된 후에 그 값이 변 경되면, 익명 메소드가 캡쳐한 변수의 값도 변경됩니다. 이것은 두 변수가 동일한 메모리에 존재하는 동일 변수이기 때문입니다. 캡쳐된 변수는 스택이 아닌 힙에 저장되게 됩니다. 익명 메소드 값은 메소드 참조 타입이며 참조 카운트가 됩니다. 지정한 익명 메소드 값에 대 한 마지막 메소드 참조가 유효 범위를 벗어나면, 혹은 제거(nil로 초기화)되거나 종료화 (finalize)되면, 그 익명 메소드가 캡쳐한 변수들은 마침내 유효 범위를 벗어나게 됩니다.


이 상황은 여러 익명 메소드가 동일한 지역 변수를 캡쳐하는 경우에 더욱 복잡해집니다. 모 든 상황에서 이 동작이 어떻게 이루어지는지 이해하려면, 익명 메소드 구현의 역학에 대해 더욱 세세하게 따져볼 필요가 있습니다.


지역 변수가 캡쳐될 때마다 그 지역 변수는 선언중인 루틴과 연관되는“프레임 객체”에 추 가됩니다. 한 루틴에서 선언된 모든 익명 메소드는 포함하는 루틴과 연관된 프레임 객체에 서 메소드로 변환됩니다. 마지막으로, 익명 메소드가 구축되어서 혹은 변수가 캡쳐되어서 생성된 모든 프레임 객체들은, 부모 프레임이 존재하고 캡쳐된 외부 변수를 액세스할 필요 가 있을 경우, 또다른 참조에 의해 부모 프레임에 연결됩니다. 하나의 프레임 객체로부터 그 부모로의 이런 연결은 레퍼런스 카운트가 됩니다. 부모 루틴으로부터 변수를 캡쳐하는 중첩 된 지역 루틴에서 선언된 익명 메소드는 그 자신이 유효범위를 벗어날 때까지 부모 프레임 객체가 유효하도록 유지합니다.

예를 들면, 다음과 같은 상황을 생각해봅시다.


anonymous_3_2.png


각 루틴과 익명 메소드에는 어떤 프레임 객체가 어디에 링크되는지를 알아보기 쉽도록 프레 임 식별자로 주석을 달았습니다.


  • v1은 F1 내의 변수입니다.
  • v2는 F1_2 내의 변수입니다. (F1_2_1에 의해 캡쳐됨) 
  • F1_1_1의 익명 메소드는 F1_1 내의 메소드입니다. 
  • F1_1은 F1에 링크됩니다. (F1_1_1이 v1을 사용함) 
  • F1_2의 익명 메소드는 F1 내의 메소드입니다. 
  • F1_2_1의 익명 메소드는 F1_2 내의 메소드입니다.


프레임 F1_2_1 및 F1_1_1은 프레임 객체를 필요로 하지 않는데, 그것은 이 프레임들은 익 명 메소드를 선언하지도 않고 캡쳐되는 변수를 갖지도 않기 때문입니다. 이 프레임들은 중 첩된 익명 메소드와 바깥의 캡쳐된 변수 사이에서 부모 지위를 가질 패스가 전혀 없습니다. (이들은 스택에 저장되는 암시적인 프레임을 갖습니다.)


익명 메소드 F1_2_1에 참조만 주어지면 변수 v1와 v2는 수명이 유지됩니다. 만약 F1의 호 출보다 더 오래 남아있는 유일한 참조가 F1_1_1이라면, 변수 v1만이 살아 남습니다. 메소드 참조/프레임 연결 체인 내에서 메모리 누수를 일으키는 순환(cycle)이 생길 수 있습 니다. 예를 들면, 익명 메소드를 직접 혹은 간접적으로 익명 메소드 자신이 캡쳐하는 변수에 저장하면 순환이 일어나게 되고, 따라서 메모리 누수가 발생됩니다.




참조링크

http://tech.devgear.co.kr/devgearbook

델파이 프로그래밍 언어 - 엠바카데로 저 | 박지훈 역 (2009년)

번호 제목 글쓴이 날짜 조회 수
공지 [DelphiCon 요약] 코드사이트 로깅 실전 활용 기법 (Real-world CodeSite Logging Techniques) 관리자 2021.01.19 14399
공지 [UX Summit 요약] 오른쪽 클릭은 옳다 (Right Click is Right) 관리자 2020.11.16 13035
공지 [10.4 시드니] What's NEW! 신기능 자세히 보기 관리자 2020.05.27 15542
공지 RAD스튜디오(델파이,C++빌더) - 고객 사례 목록 관리자 2018.10.23 21076
공지 [데브기어 컨설팅] 모바일 앱 & 업그레이드 마이그레이션 [1] 관리자 2017.02.06 22305
공지 [전체 목록] 이 달의 기술자료 & 기술레터 관리자 2017.02.06 17951
공지 RAD스튜디오(델파이, C++빌더) - 시작하기 [1] 관리자 2015.06.30 38238
공지 RAD스튜디오(델파이,C++빌더) - 모바일 앱 개발 사례 (2020년 11월 업데이트 됨) 험프리 2014.01.16 173753
583 (작성 중) 파이어몽키 모바일 개발에 대한 FAQ [1] 험프리 2015.12.01 494
582 이 달의 기술자료 - 2015년 12월 file 험프리 2015.11.27 523
581 [업데이트][10 시애틀] RAD Studio 10 시애틀 서브스크립션 업데이트 1 [3] file 험프리 2015.11.25 1568
580 [발표자료] 20151124 온라인세미나: 델파이 코드 마이그레이션 Step by Step 관리자 2015.11.25 576
579 BPL(패키지)를 이용해 프로그램 모듈화 하기 험프리 2015.11.24 1122
578 [발표자료] 20151119 델파이 소스코드의 재발견 관리자 2015.11.20 615
577 [마이그레이션][팁] 유니코드 검토 대상 분석 도구 다운로드 [1] file 험프리 2015.11.16 1838
576 [마이그레이션][팁] 설치된 컴포넌트 확인 팁 file 험프리 2015.11.12 776
575 [마이그레이션][팁] 배치파일(커맨드 명령어)을 이용해 손쉽게 소스파일 갯수를 파악할 수 있습니다. file 험프리 2015.11.12 2630
574 [안드로이드] 배포파일, 아이콘, 스플래쉬 등이 잘 배포되었는지 확인하기 file 험프리 2015.11.11 1520
573 이 달의 기술자료 - 2015년 11월 file 험프리 2015.10.30 740
572 [발표자료] 20151019 David I 초청 워크샵: 윈도우 애플리케이션 개발자를 위한 DeepDive! 관리자 2015.10.20 588
571 [시애틀] 블루투스 LE를 지원하는 디바이스에서 비콘 광고데이터를 발생할 수 있습니다. file Humphery 2015.10.06 687
570 [시애틀] 모든 종류의 안드로이드 인텐트(Intent)를 처리할 수 있습니다. Humphery 2015.10.06 847
569 [시애틀] BSON(Binary JSON)을 처리하고, JSON 데이터를 스트리밍 모델로 읽고, 쓸수 있습니다. file Humphery 2015.10.05 1714
568 [시애틀] IDE 메모리 개선으로 더 큰 프로젝트를 더 빠르게 코딩하고, 더 빠르게 컴파일 할 수 있습니다. 험프리 2015.10.05 440
567 [시애틀] 작업 중 비정상 종료 시 저장하지 않은 작업을 복구할 수 있습니다. file Humphery 2015.10.05 1012
566 [시애틀][VCL] 고해상도 모니터(4K 모니터와)와 멀티 모니터(모니터 당 DPI)를 지원합니다. [1] Humphery 2015.10.05 1298
565 [따라하기] 건강데이터 수집 및 기록 시스템 #1 - BLE 기반 스마트 체중계에서 실시간 데이터 받기 [1] Humphery 2015.10.02 5769
564 [다시보기] RAD Studio 10 시애틀 딥다이브 세션 다시보기(영문) Humphery 2015.10.02 549