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

 "델파이 프로그래밍 언어"(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 17954
공지 [UX Summit 요약] 오른쪽 클릭은 옳다 (Right Click is Right) 관리자 2020.11.16 16286
공지 [10.4 시드니] What's NEW! 신기능 자세히 보기 관리자 2020.05.27 18924
공지 RAD스튜디오(델파이,C++빌더) - 고객 사례 목록 관리자 2018.10.23 24540
공지 [데브기어 컨설팅] 모바일 앱 & 업그레이드 마이그레이션 [1] 관리자 2017.02.06 25903
공지 [전체 목록] 이 달의 기술자료 & 기술레터 관리자 2017.02.06 21275
공지 RAD스튜디오(델파이, C++빌더) - 시작하기 [1] 관리자 2015.06.30 41859
공지 RAD스튜디오(델파이,C++빌더) - 모바일 앱 개발 사례 (2020년 11월 업데이트 됨) 험프리 2014.01.16 177153
83 이 달의 기술자료 - 2020년 05월 file 험프리 2020.04.24 357
82 [발표자료] 20180621 부산 세미나-What's NEW!! RAD스튜디오: 최신 OS지원과 플랫폼 확장 방안 file 관리자 2018.06.25 357
81 [델파이 안드로이드 64비트 오류 수정] TInAppPurchase 컴포넌트 수정 file 김원경 2020.03.18 356
80 High-DPI를 적용한 윈도우 10으로 빠르게 마이그레이션 하기 관리자 2020.01.07 356
79 [고객사례-유틸리티, C++빌더] Reg Organizer - 시스템 레지스트리 구성&최적화 유틸리티 관리자 2020.01.20 355
78 [엠바카데로 Feature Friday][개발팁] 윈도우 10에서 VCL로 높은 DPI 애플리케이션 개발하기 관리자 2019.06.28 355
77 RAD Studio 10.3.3 안드로이드 앱에서 센서 컴포넌트 사용시 링크 오류 해결방법 김원경 2020.03.18 349
76 개발. 공유. 영감. – 엠바카데로 총괄 매니저가 전하는 메세지 (2020.11) 관리자 2020.12.01 348
75 [개발팁] 오래된 레거시 델파이, C++ 애플리케이션을 최신 스타일의 초고속 앱으로 마이그레이션하기 관리자 2020.10.15 348
74 이 달의 기술자료 - 2019년 08월 험프리 2019.07.26 347
73 AtoZed – IntraWeb 15 Personal 관리자 2020.09.01 346
72 [DelphiFeeds.com 리뉴얼] 새롭게, 델파이 개발자들에게 매우 유용한 뉴스 피드를 모아서 제공합니다. 관리자 2020.11.19 345
71 이 달의 기술자료 - 2020년 11월 file 험프리 2020.10.27 344
70 [베를린] GetIt 속성값을 별도의 옵션창에서 관리할 수 있습니다. 관리자 2016.05.30 344
69 RAD Studio IDE에 툴 추가하기 file 김원경 2020.03.17 343
68 [10.4 시드니][업데이트 1] 새로운 IDE 기능들 관리자 2020.10.08 341
67 [UX Summit 요약] 애플리케이션을 개발할 때 데스크톱부터 개발해야 하는 이유 (Why Desktop First to Develop an Application) 관리자 2020.11.17 340
66 [다시보기] 전세계 개발자를 위한 컨퍼런스! CodeRage 2018 file 관리자 2018.11.29 340
65 [발표자료] 현대화 vs. 소멸, 여러분의 프로젝트는 어디로 가고 있나요? 관리자 2017.08.17 340
64 [베를린] Eddystone 비콘을 사용할 수 있습니다. 관리자 2016.05.30 340