Reading Time: 2 minutesExecuting statements dynamically with compile(), exec() & eval() in Python
Executing statements dynamically with compile(), exec() & eval() in Python
Python offers two builtin functions to execute pieces of code put together as a string: eval() & exec(). These functions can be used to execute command inputs from a user, like in a custom interpreter. Optionally, these pieces of code which are in the form of a string, can be fed to the builtin compile() function first, to create a code object (Python bytecode), which can then be handed over to eval() & exec() for execution.
Let's set the pretext to our discussion today.
Help on built - in function eval in module builtins: |
eval (source[, globals [, locals ]]) - > value |
Evaluate the source in the context of globals and locals . |
The source may be a string representing a Python expression |
or a code object as returned by compile (). |
The globals must be a dictionary and locals can be any mapping, |
defaulting to the current globals and locals . |
If only globals is given, locals defaults to it. |
Help on built - in function exec in module builtins: |
exec ( object [, globals [, locals ]]) |
Read and execute code from an object , which can be a string or a code |
The globals and locals are dictionaries, defaulting to the current |
globals and locals . If only globals is given, locals defaults to it. |
Help on built - in function compile in module builtins: |
compile (source, filename, mode[, flags[, dont_inherit]]) - > code object |
Compile the source (a Python module, statement or expression) |
into a code object that can be executed by exec () or eval (). |
The filename will be used for run - time error messages. |
The mode must be 'exec' to compile a module, 'single' to compile a |
single (interactive) statement, or 'eval' to compile an expression. |
The flags argument, if present, controls which future statements influence |
the compilation of the code. |
The dont_inherit argument, if non - zero, stops the compilation inheriting |
the effects of any future statements in effect in the code calling |
compile ; if absent or zero these statements do influence the compilation, |
in addition to any features explicitly specified. |
So, the builtin functions eval() and exec() are both used to execute Python statements dynamically. These functions have 2 similarities and 2 differences. Similarities:
- Both the eval() and exec() evaluate Python statements, which can be in the form of a string, or a code object as returned by the compile() function.
- Both the eval() and exec() take two optional arguments: globals and locals.
These functions differ in the following aspects:
- eval() returns the result of the expression, while exec() does not.
- eval() only evaluates a single expression (anything on the right hand side of an assignment operation), whereas the exec() can take code blocks having loops, try/except, def clauses.
Let's take a look at basic usage of these functions:
Expand the following code snippet for more examples.
06 | Traceback (most recent call last): |
08 | File "<string>" , line 1 |
11 | SyntaxError: invalid syntax |
14 | >>> eval ( 'def anyFunction(): print(50)' ) |
15 | Traceback (most recent call last): |
16 | eval ( 'def anyFunction(): print(50)' ) |
17 | File "<string>" , line 1 |
18 | def anyFunction(): print ( 50 ) |
20 | SyntaxError: invalid syntax |
26 | >>> eval ( 'if 1: print("Hi")' ) |
27 | Traceback (most recent call last): |
28 | eval ( 'if 1: print("Hi")' ) |
29 | File "<string>" , line 1 |
32 | SyntaxError: invalid syntax |
41 | >>> exec ( 'def anyFunction(): print(50)' ) |
45 | >>> exec ( 'print(1) \nprint(2)' ) |
49 | >>> exec ( 'if 1: print("Hi")' ) |
In addition to the string form, the statements being passed to exec() & eval() can be in the form of a code object. The compile() function compiles the provided module/statement/expression, and creates a code object (contains Python bytecode) which can be passed to exec() & eval(). This is particularly useful when the same piece of code is being evaluated repeatedly.
Help on built - in function compile in module builtins: |
compile (source, filename, mode[, flags[, dont_inherit]]) - > code object |
Compile the source (a Python module, statement or expression) |
into a code object that can be executed by exec () or eval (). |
The filename will be used for run - time error messages. |
The mode must be 'exec' to compile a module, 'single' to compile a |
single (interactive) statement, or 'eval' to compile an expression. |
The flags argument, if present, controls which future statements influence |
the compilation of the code. |
The dont_inherit argument, if non - zero, stops the compilation inheriting |
the effects of any future statements in effect in the code calling |
compile ; if absent or zero these statements do influence the compilation, |
in addition to any features explicitly specified. |
Let's talk about the positional arguments.
- source: module/statement/expression.
- filename: used for run-time error messages.
- mode: 'exec'/'single'/'eval'.
Getting the filename argument out of the way of the discussion, the filename argument should give the file from which the code was read. The string entered in this argument can be retrieved using the co_filename attribute of the resultant code object. In case the source is not a part of any file, it is common practice to pass some recognizable value (such as ''). It can also be left as an empty string.
The source & mode arguments are inter-dependent, the following examples will demonstrate this.
>>> evalCodeObject = compile ( 'a + 10' , '<string>' , 'eval' ) |
>>> evaluatedValueOfa = eval (evalCodeObject) |
>>> execCodeObject = compile ( 'a = 8; a = a + 10; print(a)' , '<string>' , 'exec' ) |
>>> executeCodeBlock = exec (execCodeObject) |
>>> singleCodeObject = compile ( 'a = 50; print(a + 4); print(a + 10)' , '<string>' , 'single' ) |
>>> executeSingleCodeObject = exec (singleCodeObject) |
>>> compiledCodeBlock = compile (codeBlock, '<string>' , 'single' ) |
>>> exec (compiledCodeBlock) |
>>> compiledCodeBlock = compile (codeBlock, '<string>' , 'single' ) |
Traceback (most recent call last): |
compiledCodeBlock = compile (codeBlock, '<string>' , 'single' ) |
SyntaxError: invalid syntax |
>>> compiledCodeBlock = compile (codeBlock, '<string>' , 'single' ) |
>>> exec (compiledCodeBlock) |
>>> compiledCodeBlock = compile (codeBlock, '<string>' , 'single' ) |
Traceback (most recent call last): |
compiledCodeBlock = compile (codeBlock, '<string>' , 'single' ) |
def functionTwo(): print ( 18 ) |
SyntaxError: invalid syntax |
Now that you know about the compile() method and its modes, I believe it is the right time to tell you that the string you pass to exec() & eval() functions becomes a call to compile(source, '', mode) with the respective mode ('exec'/'eval'), which returns the code object (containing Python bytecode), which is finally executed.
Also, it is possible to have the eval() process several statements if it is in the form of a code object.
>>> compiledCodeBlock = compile (codeBlock, '<string>' , 'exec' ) |
>>> eval (compiledCodeBlock) |
SyntaxError: invalid syntax |
That's it for this one. I hope that this article helped you wrap your head around the exec(), compile() & eval() functions. Till next time!
Further Reading
See also: