Microsoft MVP성태의 닷넷 이야기
스크립트: 62. 파이썬 - class의 정적 함수를 동적으로 교체 [링크 복사], [링크+제목 복사],
조회: 10955
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

파이썬 - class의 정적 함수를 동적으로 교체

지난 글에서,

파이썬 3.x에서의 동적 함수 추가
; https://www.sysnet.pe.kr/2/0/13380

동적으로 외부의 함수를 내부에 추가하는 방법을 알아봤는데요, 이번에는 이미 정의된 함수를 대체하는 방법을 알아보겠습니다. 우선, 예제를 볼까요? ^^

class MyObject:

    def __init__(self):
        pass

    @staticmethod
    def set_name(text):
        MyObject.name = text


current_module = __import__(__name__)

class_type = current_module.__dict__.get('MyObject')
print(class_type, type(class_type))

""" 출력 결과
<class '__main__.MyObject'> <class 'type'>
"""

이렇게 구한 class_type은 "MyObject" 이름으로 접근한 것과 완전히 동일한 타입 인스턴스입니다. 그렇기 때문에 다음과 같이 멤버를 접근하는 것도 가능합니다.

class_type.set_name('hello world')  # MyObject.set_name 호출과 동일
print(MyObject.name)  # 출력 결과: hello world

그런데, 저 set_name 정적 메서드를 동적으로 접근하려면 어떻게 해야 할까요?

set_name_method = class_type.__dict__.get('set_name')
print(set_name_method, type(set_name_method))

""" 출력 결과
<staticmethod object at 0x000001B55CE77DC0> <class 'staticmethod'>
"""

__dict__ 통해 얻은 경우 예상했던 대로 (decorator로 지정한) staticmethod 타입의 인스턴스가 나오는데요, 따라서 이렇게 얻은 set_name_method로는 호출이 안 됩니다.

set_name_method('hello world')  # 오류 발생 TypeError: 'staticmethod' object is not callable

지난 글에서 다룬 staticmethod의 사용법과 비교해 보면 재미있는 결과를 얻게 되는데요,

def f_static(arg):
    print('f_static', arg)

MyObject.fs = staticmethod(f_static)
MyObject.fs('hello world')  # 정상적으로 호출
print(MyObject.fs, type(MyObject.fs))  # <function f_static at 0x000001DA86207280> <class 'function'>

위의 코드에서도 f_static을 staticmethod 타입으로 감쌌지만 MyObject.fs의 타입이 "staticmethod"가 아닌 function으로 나온다는 차이점이 있습니다.

어쨌든, (__dict__를 통해) 동적으로 구한 staticmethod 인스턴스는 decorator가 지정된 함수를 반환하는 __func__을 통해야만 합니다.

set_name_method.__func__('hello world')
print(MyObject.name)  # 출력 결과: hello world




그렇다면, 위와 같이 정의된 정적 함수를 아래의 (클래스가 아닌 모듈에 속한) 함수로,

name = None

def global_set_name(text):
    global name
    name = text

교체하려면 어떻게 해야 할까요? 간단하게는 class 이름 또는 그것의 type 인스턴스를 접근해 바꿀 수 있습니다.

# 클래스 이름을 사용해도 되고,
MyObject.set_name = global_set_name
MyObject.set_name('hello world-1')
print(name)  # 출력 결과: hello world-1

# 동적으로 구한 class 'type'의 인스턴스를 이용해도 됨
class_type.set_name = global_set_name
class_type.set_name('hello world-2')
print(name)  # 출력 결과: hello world-2

그런데, 'set_name' 함수를 (__dict__를 통해) 동적으로 접근하고 싶은 경우라면 사정이 달라집니다. 해당 함수를 구하는 경우에는 __dict__를 이용할 수 있었지만, 이걸로는 교체까진 되지 않습니다.

class_type.__dict__['set_name'] = global_set_name

# 오류 발생: TypeError: 'mappingproxy' object does not support item assignment

class_type.__dict__는 mappingproxy 타입으로 read-only라서 저런 오류가 발생합니다. 이에 대해 검색해 보면,

How to modify class __dict__ (a mappingproxy)? [duplicate]
; https://stackoverflow.com/questions/20019333/how-to-modify-class-dict-a-mappingproxy

파이썬의 setattr 내장 함수를 이용하라고 하는군요. ^^

setattr(class_type, 'set_name', global_set_name)
class_type.set_name('hello world')
print(name)  # global_set_name 함수로 교체됐으므로 전역 name 변수의 값이 바뀜
             # 출력 결과: hello world




그 외 class의 instance 함수는 decorator가 지정되지 않은 유형이므로 __func__ 등을 고려하지 않아도 된다는 점만 빼고는 위의 원칙을 그대로 적용할 수 있습니다.

또한, setattr의 두 번째 인자가 멤버를 구분하는 데 있어 키 타입으로 str 문자열을 사용한다는 점에서 파이썬에서는 같은 이름의 다른 인자를 갖는 메서드 오버로딩이 지원되지 않음을 짐작할 수 있을 것입니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 11/1/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 151  152  153  154  155  [156]  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1152정성태10/20/201123800.NET Framework: 251. string.GetHashCode는 hash 값을 cache 할까?
1151정성태10/18/201122711Java: 13. 자바도 64비트에서 (2GB) OutOfMemoryException 예외가 발생할까?
1150정성태10/18/201129985.NET Framework: 250. WPF - ComboBox의 SelectionChagned 이벤트파일 다운로드1
1149정성태10/16/201125610.NET Framework: 249. WPF - d:DesignHeight 값을 구할 수 있을까?
1148정성태10/14/201131588Java: 12. 자바에서 LINQ 사용? [7]
1147정성태10/13/201127534.NET Framework: 248. 닷넷에서 지원되는 문자열 인코딩 이름 목록
1146정성태10/12/201133239.NET Framework: 247. LINQ에서의 Max 기능 구현 [10]파일 다운로드1
1144정성태10/10/201128921.NET Framework: 246. WCF - 서버 측에서의 유효한 Timeout 설정파일 다운로드1
1143정성태10/9/201134488.NET Framework: 245. ASP.NET 서버 측 코드에서 페이스북 계정 연동하는 방법
1142정성태10/8/201134989.NET Framework: 244. 윈도우 폼을 열고 닫는 것만으로 메모리 leak이 발생할까? [2]파일 다운로드1
1141정성태10/7/201133619.NET Framework: 243. DataTable에 대해서 Dispose 메서드를 호출할 필요가 있을까? [4]파일 다운로드1
1140정성태10/6/201126784.NET Framework: 242. 닷넷 개발자 입장에서 이해해 보는 자바의 서블릿, JSP
1138정성태10/1/201144922Java: 11. 웹 로직에서 MS-SQL 서버 연결 [2]
1137정성태9/30/201130011Java: 10. 닷넷 개발자가 설치해 본 Oracle WebLogic Server - 설치 및 기본 도메인 구성
1136정성태9/29/201125434개발 환경 구성: 131. Visual Studio - ASP.NET의 Code-behind처럼 cs 파일을 그룹핑하는 매크로 함수 [2]파일 다운로드1
1135정성태9/29/201122875오류 유형: 138. TF10216: Team Foundation services are currently unavailable
1134정성태9/27/201130314.NET Framework: 241. C# 5.0에 새로 추가된 Caller Info 특성 [5]
1133정성태9/25/201133583VC++: 54. C++로 만든 WinRT 프로그램 [2]
1132정성태9/24/201173066Java: 9. 자바의 keytool.exe 사용법과 Tomcat의 SSL 통신 설정
1131정성태9/23/201129266Java: 8. 닷넷 개발자가 구현해 본 자바 웹 서비스 (2)
1130정성태9/23/201137258Java: 7. 닷넷 개발자가 구현해 본 자바 웹 서비스 (1)파일 다운로드2
1129정성태9/22/201128898개발 환경 구성: 130. Hyper-V에 MS-DOS VM 만드는 방법 - MSDN 구독자 대상 [3]
1128정성태9/20/201129073오류 유형: 137. KB2449742 보안 업데이트로 인한 충돌 문제 해결 - 두 번째 이야기
1127정성태9/19/201133164Java: 6. Java에서 MySQL 사용 [2]
1126정성태9/18/201128277Math: 3. "유클리드 호제법"과 "Bezout's identity" 구현 코드(C#)파일 다운로드1
1125정성태9/17/201126202Windows: 54. Windows 8 개발자 Preview를 사용해 보고... [2]
... 151  152  153  154  155  [156]  157  158  159  160  161  162  163  164  165  ...