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

 "델파이 프로그래밍 언어"(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 15441
공지 [UX Summit 요약] 오른쪽 클릭은 옳다 (Right Click is Right) 관리자 2020.11.16 13962
공지 [10.4 시드니] What's NEW! 신기능 자세히 보기 관리자 2020.05.27 16499
공지 RAD스튜디오(델파이,C++빌더) - 고객 사례 목록 관리자 2018.10.23 22055
공지 [데브기어 컨설팅] 모바일 앱 & 업그레이드 마이그레이션 [1] 관리자 2017.02.06 23268
공지 [전체 목록] 이 달의 기술자료 & 기술레터 관리자 2017.02.06 18923
공지 RAD스튜디오(델파이, C++빌더) - 시작하기 [1] 관리자 2015.06.30 39257
공지 RAD스튜디오(델파이,C++빌더) - 모바일 앱 개발 사례 (2020년 11월 업데이트 됨) 험프리 2014.01.16 174713
503 [발표자료] RAD Studio XE8 출시 세미나 관리자 2015.05.01 612
502 [업데이트][핫픽스][XE8] iOS 플랫폼의 SQLite 라이브러리 누락 패치 file Humphery 2015.04.30 859
501 [XE8] 번거로운 수작업을 줄여 핵심기능 개발에 집중할 수 있는 개발 생산성 향상도구 Humphery 2015.04.28 897
500 [XE8] TAppAnalytics 컴포넌트로 앱의 사용량 수집, 분석해 사용성을 이해할 수 있습니다. file Humphery 2015.04.27 803
499 델파이에서 MSOffice 엑셀과 워드 연동 참고자료. [1] Humphery 2015.04.27 3772
498 이 달의 기술자료 - 2015년 05월 file 험프리 2015.04.24 5494
497 [델파이7 이후 새로운 기능][XE2~] 프로젝트 매니저에서 타겟플랫폼을 64-bit Windows로 선택 해 64비트 애플리케이션을 개발할 수 있습니다. file Humphery 2015.04.23 4723
496 [델파이7 이후 새로운 기능] 추가되고 개선된 VCL 컴포넌트 Humphery 2015.04.22 3576
495 [델파이7 이후 새로운 기능] 코딩 생산성을 높여주는 코드에디터 기능들 Humphery 2015.04.21 2692
494 XE8로 구현한 멀티플랫폼 클립보드 공유앱 런칭 file c2design 2015.04.19 1185
493 Thalmic Labs의 MYO Armband Delphi SDK Beta7입니다. 쭈니아빠 2015.04.18 1135
492 [XE8] 오프라인에서 도움말을 볼 수 있습니다.(CHM 형식) Humphery 2015.04.17 926
491 [XE8] 대화형 지도 컴포넌트로 구글맵과 애플 맵킷 한번에 사용하기 Humphery 2015.04.17 1650
490 [XE8] 멀티-디바이스 미리보기: 다양한 디바이스 화면을 미리보며 최적화된 화면 개발하기 Humphery 2015.04.17 1132
489 XE8 새로운 기능외의 개선사항 Humphery 2015.04.16 925
488 [XE8] 버전 컨트롤 시스템 IDE 통합(Mercurial 지원) [1] Humphery 2015.04.15 1376
487 [XE8] 근거리 위치기반 서비스를 개발할 수 있는 비콘(Beacon) 연동하기 [1] Humphery 2015.04.15 3941
486 [업데이트][XE7] iOS 8.1.3/8.2 서명과 프로비저닝 프로파일 지원 핫픽스(베타) Humphery 2015.04.14 1055
485 파이어몽키에서 외부 라이브러리 연동하기(jar, so, a) [3] Humphery 2015.04.14 3545
484 [따라하기] reFind 도구를 이용해 BDE 프로젝트를 FireDAC으로 마이그레이션 따라하기 [9] Humphery 2015.04.14 4146