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

 "델파이 프로그래밍 언어"(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
623 [추가된 문법 정리] - 클래스 메소드(Class Method):델파이 2007 추가 관리자 2016.06.03 1280
622 [추가된 문법 정리] - 클래스 필드(Class Field):델파이 2007 추가 관리자 2016.06.03 828
621 [발표자료] 20160602 Upgrade VCL! 오래된 프로그램, 최신식으로 탈바꿈하기 관리자 2016.06.02 789
620 [베를린] GetIt 속성값을 별도의 옵션창에서 관리할 수 있습니다. 관리자 2016.05.30 336
619 [베를린] Eddystone 비콘을 사용할 수 있습니다. 관리자 2016.05.30 333
618 [베를린] 윈도우10에서 블루투스LE와 비콘을 활용할 수 있습니다. 관리자 2016.05.30 980
617 이 달의 기술자료 - 2016년 06월 file 험프리 2016.05.27 632
616 [업데이트][핫픽스][10.1 베를린] 갤럭시 S7 Edge(안드로이드 6.0.1) 디버깅이 되지않는 이슈 패치 험프리 2016.05.27 676
615 10.1 베를린 부터는 리본 컨트롤을 겟잇 패키지 매니저에서 설치할 수 있습니다. 험프리 2016.05.27 514
614 코드사이트(CodeSite)로 로그를 기록하며 프로그램의 문제를 파악할 수 있습니다. file 험프리 2016.05.26 2690
613 브레이크 포인트를 설정해도 디버깅이 안되는 경우 살펴볼 항목 file 험프리 2016.05.13 2808
612 [발표자료] 20160504 새로 강화된 기능들: RAD Studio, Delphi, C++Builder 관리자 2016.05.09 486
611 [베를린] TBufferedFileStream을 이용해 TFileStream 보다 더 빠르게 파일을 읽고, 쓸수 있습니다. file 험프리 2016.05.09 3084
610 [필독] Berlin Dialog 사용법 상당부분 변경 및 기능추가 (소스링크추가) [1] c2design 2016.05.04 858
609 [업데이트][핫픽스][10.1 베를린] 데이터스냅(DataSnap) ApplyUpdates 핫픽스 험프리 2016.04.28 870
608 이 달의 기술자료 - 2016년 05월 file 험프리 2016.04.26 658
607 [베를린] 스타일 디자이너에 미리보기, 클립보드 기능등이 강화되어 더욱 사용하기 편리해 졌습니다. file 험프리 2016.04.26 853
606 [베를린] 연락처 컴포넌트(TAddressBook)로 연락처 정보를 읽고 쓸수 있습니다. file 험프리 2016.04.25 620
605 [베를린] ListView 레이아웃을 입맛에 맞게 변경할 수 있습니다. file 험프리 2016.04.22 2125
604 [베를린] FireUI 앱 미리보기 - 개발중인 앱의 디자인을 장치에서 실시간으로 확인하며 개발할 수 있습니다. file 험프리 2016.04.22 1119