다음과 같은 람다 식을 작성하는 방법
def x():
raise Exception()
다음은 허용되지 않습니다 :
y = lambda : raise Exception()
답변
파이썬을 스키닝하는 방법은 여러 가지가 있습니다 :
y = lambda: (_ for _ in ()).throw(Exception('foobar'))
람다는 진술을 받아들입니다. raise ex
성명서 이므로 범용 모금자를 작성할 수 있습니다.
def raise_(ex):
raise ex
y = lambda: raise_(Exception('foobar'))
그러나 귀하의 목표가를 피하는 것이라면 def
분명히 자르지 않습니다. 그러나 다음과 같이 조건부로 예외를 발생시킬 수 있습니다.
y = lambda x: 2*x if x < 10 else raise_(Exception('foobar'))
또는 명명 된 함수를 정의하지 않고 예외를 발생시킬 수 있습니다. 당신이 필요로하는 것은 강한 위 (및 주어진 코드의 2.x)입니다.
type(lambda:0)(type((lambda:0).func_code)(
1,1,1,67,'|\0\0\202\1\0',(),(),('x',),'','',1,''),{}
)(Exception())
그리고 python3 강한 위 솔루션 :
type(lambda: 0)(type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b''),{}
)(Exception())
어떤 예외가 발생했는지 신경 쓰지 않으면 매우 간단한 답변을 지적 해 주신 @WarrenSpencer에게 감사드립니다 : y = lambda: 1/0
.
답변
어때요?
lambda x: exec('raise(Exception(x))')
답변
실제로, 방법이 있지만, 그것은 매우 고안되었습니다.
compile()
내장 함수를 사용하여 코드 객체를 만들 수 있습니다 . 이를 통해 raise
명령문 (또는 그 문제에 대한 다른 명령문)을 사용할 수 있지만 코드 오브젝트 실행이라는 또 다른 문제가 발생합니다. 일반적인 방법은 exec
문 을 사용하는 것이지만 원래 문제로 돌아갑니다. 즉, lambda
(또는 eval()
그 문제에 대해서는 문을 실행할 수 없음 )
해결책은 해킹입니다. lambda
명령문 의 결과와 같은 호출 가능 항목에는 모두 __code__
대체 할 수 있는 속성 이 있습니다. 따라서 호출 가능을 작성하고 __code__
값을 위에서 코드 오브젝트로 바꾸면 명령문을 사용하지 않고 평가할 수있는 것을 얻을 수 있습니다. 그러나이 모든 것을 달성하면 코드가 매우 모호해집니다.
map(lambda x, y, z: x.__setattr__(y, z) or x, [lambda: 0], ["__code__"], [compile("raise Exception", "", "single"])[0]()
위의 내용은 다음과 같습니다.
-
compile()
호는 예외를 발생하는 부호 객체를 생성; -
lambda: 0
반환 아무것도하지 않는다하지만 0의 값을 돌려 호출 -이 나중에 위의 코드 개체를 실행하는 데 사용됩니다; -
는 나머지 인수와 함께 첫 번째 인수
lambda x, y, z
의__setattr__
메소드 를 호출하고 첫 번째 인수를 반환 하는 함수를 작성합니다 ! 때문에, 필요__setattr__
자체가 반환None
; -
map()
호출의 결과를 소요lambda: 0
하고 사용lambda x, y, z
을 대체 그것의__code__
의 결과에 객체compile()
호출. 이 맵 연산의 결과는에 의해 리턴 된 하나의 항목이있는 목록lambda x, y, z
입니다. 이것이 우리가 이것을 필요로하는 이유입니다. 우리가 즉시lambda
사용__setattr__
하면lambda: 0
객체에 대한 참조를 잃게됩니다 ! -
마지막으로,
map()
호출에 의해 리턴 된 목록의 첫 번째 (및 유일한) 요소 가 실행되어 코드 오브젝트가 호출되어 궁극적으로 원하는 예외가 발생합니다.
작동하지만 (파이썬 2.6에서 테스트 됨) 확실히 그렇지는 않습니다.
마지막 참고 사항 : types
모듈에 액세스 할 수있는 경우 ( import
앞에 명령문 을 사용해야하는 경우 eval
)이 코드를 약간 줄일 types.FunctionType()
수 있습니다 .이를 사용 하면 주어진 코드 객체를 실행하는 함수를 만들 수 있으므로 더미 함수를 만들고 속성 lambda: 0
값을 바꾸는 해킹이 필요하지 않습니다 __code__
.
답변
람다 양식으로 작성된 함수 는 명령문을 포함 할 수 없습니다 .
답변
원하는 모든 것이 임의의 예외를 발생시키는 람다 식이면 잘못된 식으로이를 수행 할 수 있습니다. 예를 들어 lambda x: [][0]
빈 목록의 첫 번째 요소에 액세스하려고하면 IndexError가 발생합니다.
참고 : 이것은 기능이 아닌 해킹입니다. 다른 사람이 보거나 사용할 수있는 (골프가 아닌) 코드에는이 코드를 사용 하지 마십시오 .
답변
Marcelo Cantos가 제공 한 답변 의 업데이트 3 에 대해 설명하고 싶습니다 .
type(lambda: 0)(type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b''),{}
)(Exception())
설명
lambda: 0
builtins.function
클래스 의 인스턴스입니다 .
type(lambda: 0)
는 IS builtins.function
클래스.
(lambda: 0).__code__
A는 code
객체. 목적은 무엇보다도 컴파일 된 바이트 코드를 보유하고 개체입니다. CPython https://github.com/python/cpython/blob/master/Include/code.h에 정의되어 있습니다 . 메소드는 https://github.com/python/cpython/blob/master/Objects/codeobject.c에서 구현됩니다 . 코드 객체에 대한 도움말을 실행할 수 있습니다.code
Help on code object:
class code(object)
| code(argcount, kwonlyargcount, nlocals, stacksize, flags, codestring,
| constants, names, varnames, filename, name, firstlineno,
| lnotab[, freevars[, cellvars]])
|
| Create a code object. Not for the faint of heart.
type((lambda: 0).__code__)
코드 클래스입니다.
우리가 말할 때
type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b'')
다음과 같은 인수를 사용하여 코드 객체의 생성자를 호출합니다.
- argcount = 1
- kwonlyargcount = 0
- nlocals = 1
- 스택 크기 = 1
- 플래그 = 67
- codestring = b ‘| \ 0 \ 202 \ 1 \ 0’
- 상수 = ()
- 이름 = ()
- varnames = ( ‘x’,)
- 파일 이름 = ”
- 이름 = ”
- firstlineno = 1
- lnotab = b ”
PyCodeObject
https://github.com/python/cpython/blob/master/Include/code.h 의 정의에서 인수의 의미에 대해 읽을 수 있습니다 . flags
인수 의 값은 67입니다 CO_OPTIMIZED | CO_NEWLOCALS | CO_NOFREE
.
가장 importand 인수는 codestring
명령어 opcode를 포함합니다. 그들이 무엇을 의미하는지 봅시다.
>>> import dis
>>> dis.dis(b'|\0\202\1\0')
0 LOAD_FAST 0 (0)
2 RAISE_VARARGS 1
4 <0>
opcode 문서는 https://docs.python.org/3.8/library/dis.html#python-bytecode-instructions 에서 찾을 수 있습니다
. 첫 번째 바이트는의 opcode LOAD_FAST
이고 두 번째 바이트는 인수입니다 (예 : 0).
LOAD_FAST(var_num)
Pushes a reference to the local co_varnames[var_num] onto the stack.
따라서 참조를 x
스택으로 푸시합니다 . 은 varnames
오직 ‘x’를 포함하는 문자열의리스트이다. 우리는 정의하는 함수의 유일한 인수를 스택에 푸시합니다.
다음 바이트는 opcode RAISE_VARARGS
이고 다음 바이트는 인수입니다. 1
RAISE_VARARGS(argc)
Raises an exception using one of the 3 forms of the raise statement, depending on the value of argc:
0: raise (re-raise previous exception)
1: raise TOS (raise exception instance or type at TOS)
2: raise TOS1 from TOS (raise exception instance or type at TOS1 with __cause__ set to TOS)
TOS는 스택의 최상위입니다. x
함수 의 첫 번째 인수 ( )를 스택에 푸시하고 argc
1
x
이므로 예외 인스턴스 인 경우를 발생 시키거나 인스턴스를 작성 x
하고 그렇지 않으면 발생시킵니다.
마지막 바이트, 즉 0은 사용되지 않습니다. 유효한 opcode가 아닙니다. 거기에 없을 수도 있습니다.
우리는 코드 스 니펫으로 돌아가서
type(lambda: 0)(type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b''),{}
)(Exception())
코드 객체의 생성자를 호출했습니다.
type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b'')
코드 객체와 빈 사전을 함수 객체의 생성자에 전달합니다.
type(lambda: 0)(type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b''),{}
)
함수 객체에 대해 help를 호출하여 인수가 무엇을 의미하는지 봅시다.
Help on class function in module builtins:
class function(object)
| function(code, globals, name=None, argdefs=None, closure=None)
|
| Create a function object.
|
| code
| a code object
| globals
| the globals dictionary
| name
| a string that overrides the name from the code object
| argdefs
| a tuple that specifies the default argument values
| closure
| a tuple that supplies the bindings for free variables
그런 다음 Exception 인스턴스를 인수로 전달하는 생성 된 함수를 호출합니다. 결과적으로 우리는 예외를 발생시키는 람다 함수를 호출했습니다. 스 니펫을 실행하고 실제로 의도 한대로 작동하는지 봅시다.
>>> type(lambda: 0)(type((lambda: 0).__code__)(
... 1,0,1,1,67,b'|\0\202\1\0',(),(),('x',),'','',1,b''),{}
... )(Exception())
Traceback (most recent call last):
File "<stdin>", line 3, in <module>
File "", line 1, in
Exception
개량
우리는 바이트 코드의 마지막 바이트가 쓸모가 없다는 것을 알았습니다. 이 복잡한 표현을 복잡하게 어지럽히 지 마십시오. 그 바이트를 제거하자. 또한 약간의 골프를 원한다면 Exception 인스턴스화를 생략하고 대신 Exception 클래스를 인수로 전달할 수 있습니다. 이러한 변경으로 인해 다음 코드가 생성됩니다.
type(lambda: 0)(type((lambda: 0).__code__)(
1,0,1,1,67,b'|\0\202\1',(),(),('x',),'','',1,b''),{}
)(Exception)
우리가 그것을 실행할 때 우리는 이전과 같은 결과를 얻을 것입니다. 더 짧아요