Statements and Expressions

A statement is an instruction that the Python interpreter can execute. We have only seen the assignment statement and the print statement so far. Assignment statements are when a value is assigned to a variable with the assignment operator, =. Print statements are when a call to the function print occurs. print is a special function because it has special rules. The most important, for now, is that print only works if it is the first thing on that particular line. For example, you will get an error from trying to run code like this: x = print(3). (Later in this book, you’ll learn more about why that is!)

Some other kinds of statements that you’ll see shortly are while statements, for statements, if statements, and import statements. (There are other kinds too!)

An expression is a combination of literals, variable names, operators, and calls to functions. Expressions need to be evaluated. The result of evaluating an expression is a value or object.

If you ask Python to print an expression, the interpreter evaluates the expression and displays the result.

In this example len is a built-in Python function that returns the number of characters in a string.

The evaluation of an expression produces a value, which is why expressions can appear on the right hand side of assignment statements. A literal all by itself is a simple expression, and so is a variable. Evaluating a variable gives the value that the variable refers to.

In a program, anywhere that a literal value (a string or a number) is acceptable, a more complicated expression is also acceptable. Here are all the kinds of expressions we’ve seen so far:

literal
e.g., “Hello” or 3.14
variable name
e.g., x or len
operator expression
<expression> operator-name <expression>
function call expressions
<expression>(<expressions separated by commas>)

Notice that operator expressions (like + and *) have sub-expressions before and after the operator. Each of these can themselves be simple or complex expressions. In that way, you can build up to having pretty complicated expressions.

Similarly, when calling a function, instead of putting a literal inside the parentheses, a complex expression can be placed inside the parentheses. (Again, we provide some hidden code that defines the functions square and sub).

With a function call, it’s even possible to have a complex expression before the left parenthesis, as long as that expression evaluates to a function object. For now, though, we will just use variable names (like square, sub, and len) that are directly bound to function objects.

It is important to start learning to read code that contains complex expressions. The Python interpreter examines any line of code and parses it into components. For example, if it sees an = symbol, it will try to treat the whole line as an assignment statement. It will expect to see a valid variable name to the left of the =, and will parse everything to the right of the = as an expression. It will try to figure out whether the right side is a literal, a variable name, an operator expression, or a function call expression. If it’s an operator expression, it will further try to parse the sub-expressions before and after the operator. And so on. You should learn to parse lines of code in the same way.

In order to evaluate an operator expression, the Python interpreter first completely evaluates the expression before the operator, then the one after. In order to evaluate a function call expression, the interpreter evaluates the expression before the parentheses (i.e., it looks up the name of the function). Then it tries to evaluate each of the expressions inside the parentheses. There may be more than one, separated by commas. The values of those expressions are passed as inputs to the function when the function is called.

If a function call expression is a sub-expression of some more complicated expression, as square(x) is in sub(square(y), square(x)), then the return value from square(x) is passed as an input to the sub function. This is one of the tricky things that you will have to get used to working out when you read (or write) code. In this example, the square function is called (twice) before the sub function is called, even though the sub function comes first when reading the code from left to right.

To start giving you some practice in reading and understanding complicated expressions, try doing the Parsons problem below. Be careful not to indent any of the lines of code; that’s something that will come later in the course.

Your job is to order the code fragments in the order in which the Python interpreter would evaluate them. x is 2 and y is 3. Now the interpreter is executing square(x + sub(square(y), 2 *x)).

        rec-5-1: look up the variable square to get the function object
---
look up the variable x to get 2
---
look up the variable sub to get the function object
---
look up the variable square, again, to get the function object
---
look up the variable y to get 3
---
run the square function on input 3, returning the value 9
---
look up the variable x, again, to get 2
---
multiply 2 * 2 to get 4
---
run the sub function, passing inputs 9 and 4, returning the value 5
---
add 2 and 5 to get 7
---
run the square function, again, on input 7, returning the value 49
        
Next Section - Order of Operations