2.1.1 Questions and Answers
2.1.1 Questions and Answers
2.1.3 Equality
2.1.4 Equality
2.1.5 Equality
This is easy, too. The answer will be (or rather, always is) False.
2.1.6 Equality
Note that we cannot find the answer if we do not know what value
is currently stored in the variable var.
If the variable has been changed many times during the execution
of your program, or its initial value is entered from the console, the
answer to this question can be given only by Python and only at
runtime.
2.1.7 Equality
BlackSheep == (2 * WhiteSheep)
You can also ask this question using the > (greater than) operator.
If you want to know if there are more black sheep than white ones,
you can write it as follows →
True confirms it; False denies it.
What can you do with the answer you get from the computer?
There are at least two possibilities: first, you can memorize it (store
it in a variable) and make use of it later. How do you do that? Well,
you would use an arbitrary variable like this →
The content of the variable will tell you the answer to the question
asked.
You already know how to ask Python questions, but you still don’t
know how to make reasonable use of the answers. You have to
have a mechanism which will allow you to do something if a
condition is met, and not do it if it isn’t.
It’s just like in real life: you do certain things or you don’t when a
specific condition is met or not, e.g., you go for a walk if the
weather is good, or stay home if it’s wet and cold.
To make such decisions, Python offers a special instruction. Due to
its nature and its application, it’s called a conditional instruction (or
conditional statement).
There are several variants of it. We’ll start with the simplest,
increasing the difficulty slowly.
The first form of a conditional statement, which you can see to the
side, is written very informally but figuratively →
the if keyword;
As you can see, making a bed, taking a shower and falling asleep
are all executed conditionally – when SheepCounter reaches the
desired limit.
Feeding the sheepdogs, however, is always done.
Now we’re going to discuss another variant of the conditional
statement, which also allows you to perform an additional action
when the condition is not met.
Now we know what we’ll do if the conditions are met, and we know
what we’ll do if not everything goes our way. In other words, we
have a “Plan B”.
Python allows us to express such alternative plans. This is done with
a second, slightly more complex form of the conditional statement –
here it is →
Our next example resembles nesting, but the similarities are very
slight. Again, we’ll change our plans and express them as follows: If
the weather is fine, we’ll go for a walk, otherwise if we get tickets,
we’ll go to the theater, otherwise if there are free tables at the
restaurant, we’ll go for lunch; if all else fails, we’ll return home and
play chess.
Have you noticed how many times we’ve used the word
“otherwise”? This is the stage where the elifkeyword plays its role.
Let’s write the same scenario using Python →
Now we’re going to show you some simple yet complete programs.
We won’t explain them in detail, because we consider the comments
inside the code to be sufficient guides.
All the programs solve the same problem – they find the largest of
several numbers and print it out. We’ll start with the simplest case –
how to identify the larger of two numbers →
This snippet should be clear – it reads two integer values, compares
them, and finds which is the larger.
Now we’re going to show you one intriguing fact.
Practice Lab
We personally think it’s ugly and misleading, and we’re not going to
use this style in our future programs, but it’s definitely worth
knowing if you want to read and understand someone else’s
programs.
There are no other differences in the code.
It’s time to complicate the code – let’s find the largest of three
numbers. Will it enlarge the code? A bit.
You should now be able to write a program which finds the largest
of four, five, six, or even ten numbers.
You already know the scheme, so extending the size of the problem
will not be particularly complex.
But what happens if we ask you to write a program that finds the
largest of two hundred numbers? Can you imagine the code?
You’ll need two hundred variables. If two hundred variables isn’t
bad enough, try to imagine searching for the largest of a million
numbers.
Imagine a code that contains 199 conditional statements and two
hundred invocations of the input() function.
Luckily, you don’t need to deal with that. There’s a simpler
approach.
We’ll ignore the requirements of Python syntax for now, and try to
analyze the problem without thinking about the real programming.
In other words, we’ll try to write the algorithm, and when we’re
happy with it, we’ll implement it.
In this case, we’ll use a kind of notation which is not an actual
programming language (it can be neither compiled nor executed),
but it is formalized, concise and readable.
It’s called pseudocode.
There’s an example of pseudocode here →
We make a deal with the user: when the value -1is entered, it will
be a sign that there are no more data and the program should end
its work.
Otherwise, if the entered value is not equal to -1, the program will
read another number, and so on.
The trick is based on the assumption that any part of the code can
be performed more than once – precisely, as many times as
needed.
Performing a certain part of the code more than once is called a
loop. The meaning of this term is probably obvious to you.
Lines 02 through 08 make a loop. We’ll pass through them as many
times as needed to review all the entered values.
Can you use a similar structure in a program written in Python?
Yes, you can.
Practice Labs
Note that this record also declares that if there is nothing to do,
nothing at all will happen.
Analyze the program carefully. Locate the loop’s body and find out
how the body is exited.
Check how this code implements the algorithm we showed you
earlier.
2.2.6 Looping your code with while
Practice Lab
Lab PCA_2_2_6_1_A Essentials of the while loop - Guess the secret
number:
Launch in Edube | Launch Sample Solution
2.2.7 Looping your code with while
Try to recall how Python interprets the truth of a condition, and note
that these two forms are equivalent →
There are some new elements. Let us tell you about them:
the for keyword opens the for loop; note – there’s no condition
after it; you don’t have to think about conditions, as they’re
checked internally, without any intervention;
any variable after the for keyword is the control variable of the
loop; it counts the loop’s turns, and does it automatically;
the in keyword introduces a syntax element describing the range
of possible values being assigned to the control variable;
the range() function (this is a very special function) is
responsible for generating all the desired values of the control
variable; in our example, the function will create (we can even
say that it will feed the loop with) subsequent values from the
following set: 0, 1, 2 .. 97, 98, 99; note: in this case,
the range() function starts its job from 0 and finishes it one step
(one integer number) before the value of its argument;
note the pass keyword inside the loop body – it does nothing at
all; it’s an empty instruction – we put it here because
the for loop’s syntax demands at least one instruction inside the
body (by the way – if, elif, else and while express the same
thing)
Our next examples will be a bit more modest in the number of loop
repetitions.
2.2.13 Looping your code with for
Note:
the loop has been executed ten times (it’s the range() function’s
argument)
the last control variable’s value is 9 (not 10, as it starts from 0,
not from 1)
In this case, the first argument determines the initial (first) value of
the control variable.
The last argument shows the first value the control variable will not
be assigned.
Note: the range() function accepts only integers as its arguments,
and generates sequences of integers.
Can you guess the output of the program?
Here it is →
The first shown value is 2 (taken from the range()’s first argument)
The last is 7 (although the range()’s second argument is 8).
Practice Lab
Lab PCA_2_2_21_1_A Essentials of the for loop - counting
mississipply:
Launch in Edube | Launch Sample Solution
Practice Lab
Lab PCA_2_2_22_1_A The break statement - stuck in a loop:
Launch in Edube | Launch Sample Solution
Practice Lab
Lab PCA_2_2_23_1_A The continue statement - the Great Vowel
Eater:
Launch in Edube | Launch Sample Solution
Both loops, while and for, have one interesting (and rarely used)
feature.
We’ll show you how it works – try to judge for yourself if it’s usable
and whether you can live without it or not.
In other words, try to convince yourself if the feature is valuable
and useful, or is just syntactic sugar.
Take a look at the snippet →
The while’s condition is False at the beginning – can you see it?
As you can see, the else branch has been executed despite that →
Here it is →
When the loop’s body isn’t executed, the control variable retains the
value it had before the loop.
Note: if the control variable doesn’t exist before the loop starts, it
won’t exist when the execution reaches the else branch.
How do you feel about this variant of else?
Now we’re going to tell you about some other kinds of variables.
Our current variables can only store one value at a time, but there
are variables that can do much more – they can store as many
values as you want.
Practice Labs
Have you noticed that the conditions we’ve used so far have been
very simple, not to say, quite primitive? The conditions we use in
real life are much more complex. Let’s look at this sentence:
If we have some free time, and the weather is good, we will go for a
walk.
We’ve used the conjunction “and”, which means that going for a
walk depends on the simultaneous fulfilment of these two
conditions. In the language of logic, such a connection of conditions
is called a conjunction. And now another example:
If you are in the mall or I am in the mall, one of us will buy a gift for
Mom.
The appearance of the word “or” means that the purchase depends
on at least one of these conditions. In logic, such a compound is
called a disjunction.
It’s clear that Python must have operators to build conjunctions and
disjunctions. Without them, the expressive power of the language
would be substantially weakened. They’re called logical operators.
2.3.5 To be or not to be
You may remember De Morgan’s laws from school. They say that:
The negation of a conjunction is the disjunction of the negations.
The negation of a disjunction is the conjunction of the negations.
Let’s write the same thing using Python →
Note how the parentheses have been used to code the expressions
– we put them there to improve readability.
We should add that none of these two-argument operators can be
used in the abbreviated form known as op=. This exception is worth
remembering.
2.3.8 Logical values vs. single bits
Let’s look at the negation operators now. First the logical one:
logneg = not i
The logneg variable will be set to False – nothing more needs to be
done.
The bitwise negation goes like this:
bitneg = ~i
It may be a bit surprising: the bitneg variable value is -16. This
may seem strange, but isn’t at all. If you wish to learn more, you
should check out the binary numeral system and the rules
governing two’s complement numbers.
We’ll now show you what you can use bitwise operators for.
Imagine that you’re a developer obliged to write an important piece
of an operating system. You’ve been told that you’re allowed to use
a variable assigned in the following way →
x & 1 = x
x & 0 = 0
If you apply the & operation to the FlagRegister variable
along with the following bit image:
0000000000000000000000000000001000
(note the 1 at your bit’s position) as the result, you obtain one
of the following bit strings:
0000000000000000000000000000001000if your bit was set to 1
0000000000000000000000000000000000if your bit was reset
to 0
Such a sequence of zeros and ones, whose task is to grab the
value or to change the selected bits, is called a bit mask. Let’s
build a bit mask to detect the state of your bit. It should point
to the third bit. That bit has the weight of 23 = 8. A suitable
mask could be created by the following declaration:
TheMask = 8
You can also make a sequence of instructions depending on the
state of your bit – here it is →
2.3.18 How do we deal with single bits?
reset your bit – you assign a zero to the bit while all the other
bits must remain unchanged; let’s use the same property of the
conjunction as before, but let’s use a slightly different mask –
exactly as below:
1111111111111111111111111111110111
Note that the mask was created as a result of the negation of all the
bits of TheMask variable. Resetting the bit is simple, and looks like
this (choose the one you like more) →
set your bit – you assign a 1 to your bit, while all the remaining
bits must remain unchanged; use the following disjunction
property:
x | 1 = 1
x | 0 = x
You’re now ready to set your bit with one of the following
instructions →
2.3.20 How do we deal with single bits?
x ^ 1 = ~x
x ^ 0 = x
and negate your bit with the following instructions →
And here is the updated priority table, containing all the operators
introduced so far →
It may happen that you have to read, store, process, and finally,
print dozens, maybe hundreds, perhaps even thousands of
numbers. What then? Do you need to create a separate variable for
each value? Will you have to spend long hours writing statements
like the one here? →
If you don’t think that this is a complicated task, then take a piece
of paper and write a program that reads five numbers, prints them
in order from the smallest to the largest (NB, this kind of processing
is called sorting). You should find that you don’t even have enough
paper to complete the task.
So far, you’ve learned how to declare variables that are able to
store exactly one given value at a time. Such variables are
sometimes called scalars by analogy with mathematics. All the
variables you’ve used so far are actually scalars.
Think of how convenient it would be to declare a variable that could
store more than one value. For example, 100 or 1000 or even
10,000. It would still be one and the same variable, but very wide
and capacious. Sounds appealing? Perhaps, but how would it handle
such a container full of different values? How would it choose just
the one you need?
What if you could just number them? And then say: give me the
value number 2; assign the value number 15; increase the value
number 10000.
We’ll show you how to declare such multi-value variables. We’ll do
this with the example we just suggested. We’ll to write a program
that sorts a sequence of numbers. We won’t be particularly
ambitious – we’ll assume that there are exactly five numbers.
2.4.2 Lists – why do we need them so much?
Let’s create a variable called numbers; it’s assigned with not just
one number, but is filled with a listconsisting of five values (note:
the list starts with an opening bracket and ends with a closing
bracket; the space between the brackets is filled with five numbers
separated by commas).
Let’s say the same thing using adequate terminology: numbers is a
list consisting of five values, all of them numbers. We can also say
that this statement creates a list of length equal to five (as in there
are five elements inside it).
The elements inside a list may have different types. Some of them
may be integers, others floats, and yet others may be lists.
Python has adopted a convention stating that the elements in a list
are always numbered starting from zero. This means that the item
stored at the beginning of the list will have the number zero. Since
there are five elements in our list, the last of them is assigned the
number four. Don’t forget this. You’ll soon get used to it, and it’ll
become second nature.
Before we go any further in our discussion, we have to state the
following: our list is a collection of elements, but each element is a
scalar.
Note: we’ll show you the current content of our list in the lower-
right corner of the slide – it should help you follow what is
happening.
And now we want the value of the fifth element to be copied to the
second element – can you guess how to do it?
The value inside the brackets which selects one element of the list is
called an index, while the operation of selecting an element from
the list is known as indexing.
Note: all the indices used so far are literals. Their values are fixed at
runtime, but any expression can be the index, too. This opens up
lots of possibilities.
The length of a list may vary during execution. New elements may
be added to the list, while others may be removed from it. This
means that the list is a very dynamic entity.
If you want to check the list’s current length, you can use a function
named len() (its name comes from ‘length’).
The function takes the list’s name as an argument, and returns the
number of elements currently stored inside the list (in other
words – the list’s length).
The following snippet will print 5 to the console.
You can’t access an element which doesn’t exist – you can neither
get its value nor assign it a value. Both of these instructions will
cause runtime errors →
Note: we’ve removed one of the list’s elements – there are only four
elements in the list now. This means that element number four
doesn’t exist.
It may look strange, but negative indices are legal, and can be very
useful.
An element with an index equal to -1 is the last one in the list.
The example snippet will output 1→
2.4.11 Negative indices are legal!
Practice Lab
Lab PCA_2_4_11_1_A The basics of lists:
Launch in Edube | Launch Sample Solution
Note: the name of the method is preceded by the name of the data
which owns the method. Next, you add a dot, followed by the
method name, and a pair of parenthesis enclosing the arguments.
The method will behave like a function, but can do something more
– it can change the internal state of the data from which it has been
invoked.
You may ask: why are we talking about methods, not about lists?
This is an essential issue right now, as we’re going to show you how
to add new elements to an existing list. This can be done with
methods owned by all the lists, not by functions.
A new element may be glued to the end of the existing list.
Such an operation is performed by a method named append(). It
takes its argument’s value and puts it at the end of the list which
owns the method.
The list’s length then increases by one.
2.4.15 Adding elements to a list
Note: the former first element is now the second, the second the
third, and so on.
2.4.17 Adding elements to a list
The following snippet inserts 333 into the list, making it the second
element →
The former second element becomes the third, the third the fourth,
and so on.
You can start a list’s life by making it empty (this is done with an
empty pair of brackets) and then adding new elements to it as
needed.
Take a look at the snippet →
You get the same sequence, but in reverse order (this is the merit
of using the insert()method) →
The for loop has a very special variant that can process lists very
effectively – let’s take a look at that.
Let’s assume that you want to calculate the sum of all the values
stored in the list list. You need a variable whose sum will be
stored and initially assigned a value of 0 – its name is sum. Then
you add to it all the elements of the list using the forloop. Take a
look at the snippet →
Let’s comment on this example:
the list is assigned a sequence of five integer values;
the i variable takes the values 0, 1, 2, 3, and 4, and then it
indexes the list, selecting the subsequent elements: the first,
second, third, fourth and fifth;
each of these elements is added together by the += operator to
the sum variable, giving the final result at the end of the loop;
note the way in which the len() function has been employed – it
makes the code independent of any possible changes in the list’s
content.
But the for loop can do much more. It can hide all the actions
connected to the list’s indexing, and deliver all the list’s elements in
a handy way.
This modified snippet shows how it works →
the for instruction specifies the variable used to browse the list
(i here) followed by the inkeyword and the name of the list
being processed (list here)
the i variable is assigned the values of all the subsequent list’s
elements, and the process occurs as many times as there are
elements in the list;
this means that you use the i variable as a copy of the elements’
values, and you don’t need to use indices;
the len() function is not needed here, either.
Let’s leave lists aside for a moment and look at one intriguing issue.
Imagine that you need to rearrange the elements of a list, i.e.,
reverse the order of the elements: the first and the fifth as well as
the second and fourth elements will be swapped. The third one will
remain untouched. Question: how can you swap the values of two
variables? Take a look at the snippet →
If you do something like this, you would lose the value previously
stored in variable2. Changing the order of the assignments will not
help. You need a third variable that serves as an auxiliary storage.
2.4.25 Lists in action
Now you can easily swap the list’s elements to reverse their order →
The snippet’s output looks like this:
[5, 3, 8, 1, 10]
It looks fine with five elements.
Will it still be acceptable with a list containing 100 elements? No, it
won’t.
Can you use the for loop to do the same thing automatically,
irrespective of the list’s length?
Yes, you can.
Note:
we’ve assigned the l variable with the current list’s length (this
makes our code a bit clearer and shorter)
we’ve launched the for loop to run through its body l//2 times
(this works well for lists with both even and odd lengths, because
when the list contains an odd number of elements, the middle
one remains untouched)
we’ve swapped the ith element (from the beginning of the list)
with the one with an index equal to (l – i – 1) (from the end
of the list); in our example, for i equal to 0 the (l – i -
1) gives 4; for i equal to 1, it gives 3 – this is exactly what we
needed.
Lists are extremely useful, and you’ll encounter them very often.
Practice Lab
Lab PCA_2_4_28_1_A The basics of lists:
Launch in Edube | Launch Sample Solution
Now that you can effectively juggle the elements of lists, it’s time to
learn how to sort them. Many sorting algorithms have been
invented so far, which differ a lot in speed, as well as in complexity.
We are going to show you a very simple algorithm, easy to
understand, but unfortunately not too efficient, either. It’s used
very rarely, and certainly not for large and extensive lists.
Let’s say that a list can be sorted in two ways:
We go further, and look at the third and the fourth elements. Again,
this is not what it’s supposed to be like. We have to swap them →
Now we check the fourth and the fifth elements. Yes, they too are in
the wrong positions. Another swap occurs →
2.5.4 Sorting simple lists
The first pass through the list is already finished. We’re still far from
finishing our job, but something curious has happened in the
meantime. The largest element, 10, has already gone to the end of
the list. Note that this is the desired place for it. All the remaining
elements form a picturesque mess, but this one is already in
place →
Now, for a moment, try to imagine the list in a slightly different way
– namely, like this →
Look – 10 is at the top. We could say that it floated up from the
bottom to the surface, just like the bubble in a glass of champagne.
The sorting method derives its name from the same observation –
it’s called a bubble sort.
Now we start with the second pass through the list. We look at the
first and second elements – a swap is necessary →
2.5.7 Sorting simple lists
Time for the third and fourth elements: we have to swap them
too →
Now 6 needs to go into place. We swap the second and the third
elements →
2.5.10 Sorting simple lists
You should be able to read and understand this program without any
problems →
It is as simple as that →
The program:
The surprising part is the fact that the program will output:
[2]
not
[1]
which seems to be the obvious solution.
Lists (and many other complex Python entities) are stored in
different ways than ordinary (scalar) variables.
You could say that:
Read these two lines once more – the difference is essential for
understanding what we are going to talk about next.
The assignment:
list2 = list1
copies the name of the array, not its contents. In effect, the
two names (list1 and list2) identify the same location in the
computer memory. Modifying one of them affects the other, and
vice versa.
How do you cope with that?
2.6.2 Slices – a powerful tool
As you can see, it resembles indexing, but the colon inside makes a
big difference.
A slice of this form makes a new (target) list, taking elements from
the source list – the elements of the indices from start to end-1.
Note: not to end but to end-1. An element with an index equal
to end is the first element which does not take part in the slicing.
Using negative values for both start and end is possible (just like in
indexing).
To repeat:
If you omit the start in your slice, it is assumed that you want to get
a slice beginning at the element with index 0.
In other words, the slice of this form:
list[:end]
is a more compact equivalent of:
list[0:end]
This is why this snippet’s output is:
[10, 8, 6]
Similarly, if you omit the end in your slice, it is assumed that you
want the slice to end at the element with the index len(list).
In other words, the slice of this form:
list[start:]
is a more compact equivalent of:
list[start:len(list)]
The snippet’s output is:
[4, 2]
Omitting both start and end makes a copy of the whole list →
The snippet’s output is:
[10, 8, 6, 4, 2]
Removing the slice from the code changes its meaning dramatically.
Take a look →
The del instruction will delete the list itself, not its content.
The print() function invocation from the last line of the code will
then cause a runtime error.
2.6.14 The in operator
Python offers two very powerful operators, able to look through the
list in order to check whether a specific value is stored inside the list
or not.
These operators are →
The first of them (in) checks if a given element (its left argument)
is currently stored somewhere inside the list (the right argument) –
the operator returns True in this case.
The second (not in) checks if a given element (its left argument) is
absent in a list – the operator returns True in this case.
Practice Lab
Note:
the elements of the rows are fields, eight of them per row;
the elements of the chessboard are rows, eight of them per
chessboard.
Access to the selected field of the board requires two indices – the
first selects the row, the second, the field number inside the row,
which is de facto a column number.
Take a look at the chessboard. Every field contains a pair of indices
which should be given to access the field’s content →
Glancing at the figure shown here → let’s set some chess pieces on
the board. First, let’s add all the rooks:
board[0][0] = ROOK
board[0][7] = ROOK
board[7][0] = ROOK
board[7][7] = ROOK
If you want to add a knight to C4, you do it as follows:
board[4][2] = KNIGHT
And now a pawn to E5:
board[3][4] = PAWN
The whole matrix is filled with zeros now. You can assume that it’s
updated automatically using special hardware agents. The thing you
have to do is to wait for the matrix to be filled with measurements.
Now find the highest temperature during the whole month – see the
code →
Note:
Now count the days when the temperature at noon was at least 20
ºC →
Python does not limit the depth of list-in-list inclusion. Here you can
see an example of a three-dimensional array.
Imagine a hotel. It’s a huge hotel consisting of three buildings, 15
floors each. There are 20 rooms on each floor. For this, you need an
array which can collect and process information on the
occupied/free rooms.
First step – the type of the array’s elements. In this case, a Boolean
value (True/False) would fit.
Step two – calm analysis of the situation. Summarize the available
information: three buildings, 15 floors, 20 rooms.
Now you can create the array →
The first index (0 through 2) selects one of the buildings; the
second (0 through 14) selects the floor, the third (0 through 19)
selects the room number. All rooms are initially free.
Now you can book a room for two newlyweds: in the second
building, on the tenth floor, room 14:
rooms[1][9][13] = True
and release the second room on the fifth floor located in the first
building:
rooms[0][4][1] = False
Check if there are any vacancies on the 15th floor of the third
building:
vacancy = 0
for roomno in range(20):
if not rooms[2][14][roomno]:
vacancy += 1
The vacancy variable contains 0 if all the rooms are occupied, or the
number of available rooms otherwise.