This is a concise Python 3 programming tutorial for people who think that reading is boring. I try to show everything with simple code examples; there are no long and complicated explanations with fancy words.
This tutorial is aimed at people with no programming experience at all or very little programming experience. If you have programmed a lot in the past using some other language you may want to read the official tutorial instead.
You can use Python 3.5 or any newer Python with this tutorial. Don't use Python 2 because it's no longer supported.
The tutorial consists of two sections:
This section will get you started with using Python and you'll be able to learn more about whatever you want after studying it.
- What is programming?
- Installing Python
- Getting started with Python
- ThinkPython: The way of the program
- Variables, Booleans and None
- Using functions
- Setting up an editor
- If, else and elif
- Handy stuff with strings
- Lists and tuples
- Loops
- Dictionaries
- Defining functions
- Modules
- Exceptions
- Classes
Feel free to skip this part if you already know everything it's talking about.
As a computer user you know that computers don't have feelings. They don't work any faster or slower depending on if we're angry at them or if we're happy. Computers can perform millions of calculations per second, but they require us to tell them exactly what to do. If they do something else than we want them to do the problem is usually that they don't understand our instructions the way we understand them.
The only big difference between programming and what you're familiar with already is that instead of clicking buttons to do things we write the instructions using a programming language. Most programming languages consist of English words, digits and some characters that have special meanings.
Unlike people often think, programming is usually not complicated. Large programs are always made of small, simple pieces, and those pieces are written one by one. Programming languages are made to be used by humans, so if there's an easy way to do something and a difficult way to do something, you should use the easier way.
First of all, you don't need to be good at math. Some programmers are good at math, some are not. Programming and math are two separate things and being good or bad at one doesn't mean you are automatically good or bad at the other.
You also don't need a powerful computer. I could do almost all of my programming on a 12-year-old computer if I needed to. Fast computers are nice to work with, but you don't need them.
Programming takes time like all hobbies do. Some people learn it quickly, and some people don't. I don't expect you to read this tutorial in a couple hours and then master everything it's talking about. Take your time with things, and remember that I learned to program slowly.
This tutorial uses a programming language called Python because it's easy to learn and we can do many different things with it. For example, we can create our own applications that have buttons that people can click instead of just using applications written by others.
Before we can get started with Python we need to know how to write some of Python's special characters with our keyboards. Unfortunately I don't know which keys you need to press to produce these characters because your keyboard is probably different than mine. But the keyboard can tell what you need to press. For example, my Finnish keyboard has a key like this:
Here's what the characters on this key mean:
- I can type a number 7 by pressing this key without holding down other keys at the same time.
- I can type a
/
character by holding down the shift key (on the left edge of the keyboard, between Ctrl and CapsLock) and pressing this key. - I can type a
{
character by holding down AltGr (on the bottom of the keyboard, on the right side of the spacebar) and pressing this key. Holding down Ctrl and Alt instead of AltGr may also work.
The only key that doesn't have anything written on it is spacebar. It's the big, wide key that's closest to you. Another key that's used for producing whitespace is tab, the key above CapsLock.
In this tutorial we need to know how to type these characters. We'll learn their meanings later.
Character | Names |
---|---|
+ |
plus |
- |
minus, dash |
_ |
underscore |
* |
star, asterisk |
/ |
forwardslash (it's leaning forward) |
\ |
backslash (it's leaning back) |
= |
equals sign |
% |
percent sign |
. |
dot |
, |
comma |
: |
colon |
? |
question mark |
! |
exclamation mark |
< > |
less-than and greater-than signs |
' " |
single quote and double quote |
# |
hashtag |
() |
parentheses |
[] |
square brackets, brackets |
{} |
curly braces, braces, curly brackets |
That may seem like many characters, but you probably know many of them already so it shouldn't be a problem.
I've done my best to make this tutorial as easy to follow as possible. Other people have commented on this and helped me improve this a lot also. But what should you do if you have a problem with the tutorial?
- Try the example code yourself.
- Read the code and the explanation for it again.
- If there's something you haven't seen before in the tutorial and it's not explained, try to find it in the previous chapters.
You are free to combine this tutorial with other learning resources. If this tutorial isn't exactly what you're looking for you don't need to stick with nothing but this. You can find another tutorial and mix the tutorials however you want as long as you make sure that you understand everything you read.
One of the most important things with learning to program is to not fear mistakes. If you make a mistake, your computer will not break in any way. You'll get an error message that tells you what's wrong and where. Even professional programmers do mistakes and get error messages all the time, and there's nothing wrong with it.
If you want to know what some piece of code in this tutorial does just try it and see. It's practically impossible to break anything accidentally with the things you will learn by reading this tutorial, so you are free to try out all the examples however you want and change them to do whatever you want.
Even though a good tutorial is an important part about learning to program, you also need to learn to make your own things. Use what you have learned, and create something with it.
This chapter is probably the most boring chapter in the whole tutorial. Other chapters contain much less text and much more code. You can also get pretty far by just reading the code, and then reading the text only if you don't understand the code.
- Now you should know what programming and programming languages are.
- You don't need to be good at math and you don't need a new computer.
- Complicated programs consist of simple pieces.
- You don't need to remember how to type different characters. Just find the character on your keyboard and press the key, holding down shift or AltGr as needed.
- Make sure you understand everything you read.
- Experiment with things freely and don't fear mistakes.
- Error messages are our friends.
- Let me know if you have trouble with this tutorial.
- Now we're ready to install Python and get started with the basics.
Launch Python
The >>>
means that Python is ready and we can enter a command. The
basic idea is really simple: we enter a command, press Enter, enter
another command, press Enter and keep going.
You probably don't know any Python commands yet. Let's see what happens if we just write something and press Enter.
>>> hello
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'hello' is not defined
>>>
Oops! That didn't work. But like I wrote in the introduction, error messages are our friends. This error message tells us what's wrong and where, and we'll learn what "name 'hello' is not defined" means later.
Maybe we can press Enter without typing anything?
>>>
>>>
>>>
>>>
That worked. How about numbers?
>>> 123
123
>>> -123
-123
>>> 3.14
3.14
>>> -12.3
-12.3
>>>
There we go, it echoes them back.
In some countries, decimal numbers are written with a comma, like 3,14
instead of 3.14
. Maybe Python knows that?
>>> 3,14
(3, 14)
>>>
We didn't get an error... but (3, 14)
is not at all what we expected!
So from now on, let's use a dot with decimal numbers, because 3.14
worked just fine. Later we'll learn what (3, 14)
is.
Comments are text that don't do anything when they're run.
They can be created by typing a #
and then some text after it,
and they are useful when our code would be hard to understand without them.
>>> 1 + 2 # can you guess what the result is?
3
>>>
Again, I put a space after the #
and multiple spaces before it just to
make things easier to read.
If we write a comment on a line with no code on it, the prompt changes
from >>>
to ...
. To be honest, I have no idea why it does that and I
think it would be better if it would just stay as >>>
. The prompt goes
back to >>>
when we press Enter again.
>>> # hello there
...
>>>
Strings are small pieces of text that we can use in our programs. We can create strings by simply writing some text in quotes.
>>> 'hello'
'hello'
>>> 'this is a test'
'this is a test'
>>>
Strings can also be written with "double quotes" instead of 'single quotes'. This is useful when we need to put quotes inside the string.
>>> "hello there"
'hello there'
>>> "it's sunny"
"it's sunny"
>>>
It's also possible to add single quotes and double quotes into the same string, but most of the time we don't need to do that so I'm not going to talk about it now.
It doesn't matter which quotes you use when the string doesn't need to contain any quotes. If you think that one of the quote types looks nicer than the other or you find it faster to type, go ahead and use that.
Strings can be joined together easily with +
or repeated with *
:
>>> "hello" + "world"
'helloworld'
>>> "hello" * 3
'hellohellohello'
>>>
Note that a #
inside a string doesn't create a comment.
>>> "strings can contain # characters"
'strings can contain # characters'
>>>
---------- WARNING: This part contains boring math. Proceed with caution. ----------
Let's type some math stuff into Python and see what it does.
>>> 17 + 3
20
>>> 17 - 3
14
>>> 17 * 3
51
>>> 17 / 3
5.666666666666667
>>>
It's working, Python just calculates the result and echoes it back.
I added a space on both sides of +
, -
, *
and /
. Everything would
work without those spaces too:
>>> 4 + 2 + 1
7
>>> 4+2+1
7
>>>
However, I recommend always adding the spaces because they make the code easier to read.
Things are calculated in the same order as in math. The parentheses (
and )
also work the same way.
>>> 1 + 2 * 3 # 2 * 3 is calculated first
7
>>> (1 + 2) * 3 # 1 + 2 is calculated first
9
>>>
You can also leave out spaces to show what's calculated first. Python ignores it, but our code will be easier to read for people.
>>> 1 + 2*3 # now it looks like 2*3 is calculated first
7
>>>
Python also supports many other kinds of calculations, but most of the time you don't need them. Actually you don't need even these calculations most of the time, but these calculations are probably enough when you need to calculate something.
- Error messages are our friends.
- We can enter any Python commands to the interactive
>>>
prompt, and it will echo back the result. +
,-
,*
and/
work in Python just like in math.- Pieces of text starting with a
#
are comments and pieces of text in quotes are strings. - You can use single quotes and double quotes however you want.
ThinkPython is a free Python book. Its first chapter is about programming in general. Read it here..
Read the glossary also, but don't study it too much. Some things in it are useful to know, but I think studying it carefully would be really boring. Also do all exercises except the last one. It's more about math skills than learning Python.
When you're done reading, read the summary below and make sure you've learned everything.
- Now you should have some kind of idea about what programming is.
- Each value has a type, and you can find that out with
type(value)
. For example, writingtype(123)
to the Python prompt does the same thing as writingint
to the prompt. - Types and classes are the same thing (in Python 3).
- Some of the most commonly used types are:
- int is short for integer.
1
and2
are integers. - float is short for floating point number.
1.0
and3.14
are floats. - str is short for string.
"hello"
and'hello'
are strings. It doesn't matter if you use 'single quotes' or "double quotes", they do the same thing in Python.
- int is short for integer.
- What happens if you use + between two strings, like
"hello" + "world"
? How about"hello" * "world"
? - What happens if you use + between a string and an integer, like
"hello" + 3
? How about"hello" * 3
? - What happens if you use + between a float and an integer, like
0.5 + 3
? How about0.5 * 3
?
Variables are easy to understand. They simply point to values.
>>> a = 1 # create a variable called a that points to 1
>>> b = 2 # create another variable
>>> a # get the value that the variable points to
1
>>> b
2
>>>
We can also change the value of a variable after setting it.
>>> a = 2 # make a point to 2 instead of 1
>>> a
2
>>>
Setting a variable to another variable gets the value of the other variable and sets the first variable to point to that value.
>>> a = 1
>>> b = a # this makes b point to 1, not a
>>> a = 5
>>> b # b didn't change when a changed
1
>>>
Trying to access a variable that is not defined creates an error message.
>>> thingy
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'thingy' is not defined
>>>
Variables are simple to understand, but there are a few details that we need to keep in mind:
- Variables always point to a value, they never point to other variables. That's why the arrows in our diagrams always go left to right.
- Multiple variables can point to the same value, but one variable cannot point to multiple values.
- The values that variables point to can point to other values also. We'll learn more about that when we'll talk about.
Variables are an important part of most programming languages, and they allow programmers to write much larger programs than they could write without variables.
Variable names are case-sensitive, like many other things in Python.
>>> thing = 1
>>> THING = 2
>>> thIng = 3
>>> thing
1
>>> THING
2
>>> thIng
3
>>>
There are also words that cannot be used as variable names
because they are reserved by Python itself and have a special meaning.
They are called keywords, and we can run help('keywords')
to see the full list if we want to.
We'll learn to use most of them later in this tutorial. Trying to use a
keyword as a variable name causes a syntax error.
>>> if = 123
File "<stdin>", line 1
if = 123
^
SyntaxError: invalid syntax
>>>
When assigning something to a variable using a =
, the right side of
the =
is always executed before the left side. This means that we can
do something with a variable on the right side, then assign the result
back to the same variable on the left side.
>>> a = 1
>>> a = a + 1
>>> a
2
>>>
To do something to a variable (for example, to add something to it) we
can also use +=
, -=
, *=
and /=
instead of +
, -
, *
and
/
. The "advanced" %=
, //=
and **=
also work.
>>> a += 2 # a = a + 2
>>> a -= 2 # a = a - 2
>>> a *= 2 # a = a * 2
>>> a /= 2 # a = a / 2
>>>
This is not limited to integers.
>>> a = 'hello'
>>> a *= 3
>>> a += 'world'
>>> a
'hellohellohelloworld'
>>>
Now we also understand why typing hello to the prompt didn't work in the beginning of this tutorial. But we can assign something to a variable called hello and then type hello:
>>> hello = 'hello there'
>>> hello
'hello there'
>>>
Variable names can be multiple characters long. They can contain uppercase characters, numbers and some other characters, but most of the time we should use simple, lowercase variable names. We can also use underscores. For example, these variable names are good:
>>> magic_number = 123
>>> greeting = "Hello World!"
>>>
Don't use variable names like this, these variables are bad:
>>> magicNumber = 3.14 # looks weird
>>> Greeting = "Hello there!" # also looks weird
>>> x = "Hello again!" # what the heck is x?
>>>
All of these variables work just fine, but other Python programmers
don't want you to use them. Most Python code doesn't use variable names
that contain UpperCase letters like magicNumber
and Greeting
, so
other people reading your code will think it looks weird if you use
them. The problem with x
is that it's too short, and people have no
idea what it is. Remember that mathematicians like figuring out what x
is, but programmers hate that.
There are two Boolean values, True and False. In Python, and in many
other programming languages, =
is assigning and ==
is comparing.
a = 1
sets a to 1, and a == 1
checks if a equals 1.
>>> a = 1
>>> a == 1
True
>>> a = 2
>>> a == 1
False
>>>
a == 1
is the same as (a == 1) == True
, but a == 1
is more
readable, so most of the time we shouldn't write == True
anywhere.
>>> a = 1
>>> a == 1
True
>>> (a == 1) == True
True
>>> a = 2
>>> a == 1
False
>>> (a == 1) == True
False
>>>
None is Python's "nothing" value. It behaves just like any other value, and it's often used as a default value for different kinds of things. Right now it might seem useless but we'll find a bunch of ways to use None later.
None's behavior on the interactive prompt might be a bit confusing at first:
>>> thingy = None
>>> thingy
>>>
That was weird! We set thingy to None, but typing thingy
didn't echo
back None.
This is because the prompt never echoes back None. That is handy, because many things result in None, and it would be annoying to see None coming up all the time.
If we want to see a None on the interactive prompt, we can use print.
>>> print(thingy)
None
>>>
Another confusing thing is that if we do something weird to None we get error messages that talk about NoneType object. The NoneType object they are talking about is always None. We'll learn more about what attributes and calling are later.
>>> None.hello # None has no attribute 'hello'
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'NoneType' object has no attribute 'hello'
>>> None() # None is not callable
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'NoneType' object is not callable
>>>
So far we've used ==
, but there are other operators also. This list
probably looks awfully long, but it's actually quite easy to learn.
Usage | Description | True examples |
---|---|---|
a == b |
a is equal to b | 1 == 1 |
a != b |
a is not equal to b | 1 != 2 |
a > b |
a is greater than b | 2 > 1 |
a >= b |
a is greater than or equal to b | 2 >= 1 , 1 >= 1 |
a < b |
a is less than b | 1 < 2 |
a <= b |
a is less than or equal to b | 1 <= 2 , 1 <= 1 |
We can also combine multiple comparisons. This table assumes that a and b are Booleans.
Usage | Description | True example |
---|---|---|
a and b |
a is True and b is True | 1 == 1 and 2 == 2 |
a or b |
a is True, b is True or they're both True | False or 1 == 1 , True or True |
not
can be used for negations. If value
is True, not value
is
False, and if value
is False, not value
is True.
There's also is
, but don't use it instead of ==
unless you know
what you are doing. We'll learn more about it later.
- Variables have a name and a value. We can create or change variables
with
name = value
. thing += stuff
does the same thing asthing = thing + stuff
.- Use lowercase variable names and remember that programmers hate figuring out what x is.
=
means assigning and==
means comparing.- True and False are Booleans. Comparing values results in a Boolean.
- None is a value that we'll find useful later. When error messages say
NoneType object
they mean None.
Now we know how to make Python show text.
>>> 'Hello!'
'Hello!'
>>>
But that includes ''
. One way to show text to the user without ''
is with the print function. In Python, printing doesn't have anything
to do with physical printers, it just means showing text on the screen.
>>> print('Hello!')
Hello!
>>>
Now we are ready for a classic example, which is also the first program in many tutorials :)
>>> print("Hello World!")
Hello World!
>>>
But what exactly is print?
Let's see what happens if we type print
without the ('Hello')
part.
>>> print
<built-in function print>
>>>
We could also type print(print)
, it would do the same thing. Python
replied to us with some information about print wrapped in <>
.
As we can see, print is a function. Functions do something when they are
called by typing their name and parentheses. Inside the
parentheses, we can pass some arguments too. In print("hello")
the
function is print
and we give it one argument, which is "hello"
.
Functions are easy to understand, They simply do something when they
are called. Functions run immediately when we call them, so the
text appears on the screen right away when we run print(something)
.
Sometimes people think that doing thingy = print('hello')
means that
Python is going to print hello every time we type thingy
. But this
is not correct! print('hello')
runs print right away, and if we
type thingy
later it's not going to run print('hello')
again.
Now we know that thingy = print('hello')
doesn't store the
print('hello')
call in a variable. But what does it do then?
>>> thingy = print('hello')
hello
>>> print(thingy) # thingy is now None
None
>>>
So doing thingy = print('hello')
set thingy
to None.
Here's what happened, explained in more detail:
- When we do
thingy = print('hello')
, the right side is processed first. print('hello')
calls the print function with the argument'hello'
.- The function runs. It shows the word hello.
- The print function returns None. All functions need to return something, and print returns None because there's no need to return anything else.
- Now the right side has been processed.
print('hello')
returned None, so we can imagine we have None instead ofprint('hello')
there, and the assignment now looks likethingy = None
. thingy
is now None.
Now we understand what a return value is. When we call the
function, Python "replaces" function(arguments)
with whatever the
function returns.
Calling a function without assigning the return value to anything (e.g.
print('hello')
instead of thingy = print('hello')
) simply throws away
the return value. The interactive >>>
prompt doesn't echo the return
value back because it's None.
Of course, thingy = print('hello')
is useless compared to print('hello')
because the print function always returns None and we can do thingy = None
without any printing.
Not all functions return None. The input function can be used for getting a string from the user.
>>> stuff = input("Enter something:")
Enter something:hello
>>> stuff
'hello'
>>>
input("Enter something:")
showed the text Enter something:
on the
screen and waited for me to type something. I typed hello and pressed
Enter. Then input returned the hello I typed as a string and it was
assigned to stuff
.
Usually we want to add a space after the :
, like this:
>>> stuff = input("Enter something: ") # now there's space between : and where i type
Enter something: hello
>>>
We can also print an empty line by calling print without any arguments.
>>> print()
>>>
In Python, \n
is a newline character. Printing a string that contains
a newline character also prints a newline:
>>> print('hello\nworld')
hello
world
>>>
If we want to print a real backslash, we need to escape it by typing two backslashes.
>>> print('hello\\nworld')
hello\nworld
>>>
We can also pass multiple arguments to the print function. We need to separate them with commas and print will add spaces between them.
>>> print("Hello", "World!")
Hello World!
>>>
Unlike with +
, the arguments don't need to be strings.
>>> print(42, "is an integer, and the value of pi is", 3.14)
42 is an integer, and the value of pi is 3.14
>>>
In the previous chapter we learned that if
is not a
valid variable name because it's a keyword.
>>> if = 123
File "<stdin>", line 1
if = 123
^
SyntaxError: invalid syntax
>>>
But print
and input
are not keywords, so can we use them as
variable names?
>>> print = "hello"
>>> print
'hello'
>>>
We can, but there's a problem. Now we can't even do our hello world!
>>> print("Hello World!")
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'str' object is not callable
>>>
The error message complains that strings aren't callable because we just
set print
to the string 'hello'
and now we're trying to call it like
a function. As you can see, this is not a good idea at all. Most
editors display built-in functions with a special
color, so you don't need to worry about doing this accidentally.
Exit out of Python and start it again, and print("Hello World!")
should work normally.
function()
calls a function without any arguments, andfunction(1, 2, 3)
calls a function with 1, 2 and 3 as arguments.- When a function is called, it does something and returns something.
function(arguments)
is "replaced" with the return value in the code that called it. For example,stuff = function()
calls a function, and then doesstuff = the_return_value
and the return value ends up in stuff.- Python comes with
print
andinput
. They are built-in functions. - Avoid variable names that conflict with built-in functions.
An editor is a program that lets us write longer programs than we can
write on the >>>
prompt. With an editor we can save the programs to files and
run them as many times as we want without writing them again.
When programmers say "editor" they don't mean programs like Microsoft Word or LibreOffice/OpenOffice Writer. These programs are for writing text documents, not for programming. Programming editors don't support things like bigger font sizes for titles or underlining bits of text, but instead they have features that are actually useful for programming, like automatically displaying different things with different colors, but also highlighting mistakes in the code, and coloring syntax.
If you are on Windows or Mac OSX you have probably noticed that your Python came with an editor called IDLE. We are not going to use it because it's lacking some important features, and most experienced programmers (including me) don't use it or recommend it.
The choice of an editor is a very personal thing. There are many editors, and most programmers have a favorite editor that they use for everything and recommend to everyone.
If you aren't sure about which editor you should use, I recommend Porcupine. It's a simple editor I wrote in Python; it lets you edit files and it doesn't have too many other featues. Install it with these instructions, and then learn to use it by writing the classic Hello World program, Then you can skip the rest of this chapter.
Note that most other editors come with settings that are not suitable for writing Python code. TODO: add a link to the old editor setup tutorial here.
Most of these editors lack some important features, they have so many features that confuse people or they aren't free. You can use these editors if you like them, but these editors are BAD for getting started with programming:
- PyCharm
- IDLE
- Emacs
- Gedit
- Nano
- NetBeans
- Notepad
- Pluma
- Spyder
- Vim
- Wingware
So far we have used the >>>
prompt for everything. But now we also
have an editor that lets us write longer programs. So why not just
always use the editor?
The >>>
prompt is meant to be used for experimenting with things. For
example, if you want to know what "hello" + 123
does, just open the
prompt and run it.
If you want to write something once and then run it many times, write the code to a file. For example, if you want to make a program that asks the user to enter a word and then echoes it back, write a program that does that in a file and run it as many times as you want to.
Note that if you write something like 'hello'
to the >>>
prompt it
echoes it back, but if you make a file that contains nothing but a
'hello'
it won't do anything when you run it. You need to use
print('hello')
instead when your code is in a file.
Now we know what True and False are.
>>> 1 == 1
True
>>> 1 == 2
False
>>>
>>> its_raining = True
>>> its_raining
True
>>>
But what if we want to execute different code depending on something?
That's when if
comes in.
>>> its_raining = True
>>> if its_raining:
... print("It's raining!")
...
It's raining!
>>> its_raining = False
>>> if its_raining:
... print("It's raining!") # nothing happens
...
>>>
The prompt changed from >>>
to ...
. It meant that Python was
expecting me to keep typing. When I was done, I just pressed Enter
twice. My code was executed and the prompt went back to >>>
.
An important thing to notice is that the line with a print is indented. You can press the tab key, or if it doesn't work just press space a few times.
But why is that if its_raining
instead of if(its_raining)
?
Earlier we learned that if
is a keyword.
>>> if = 123
File "<stdin>", line 1
if = 123
^
SyntaxError: invalid syntax
>>>
Functions like print
need ()
after their name to work. But if
is a keyword, not a function, so it doesn't need ()
. Python has
separate functions and keywords because it's possible to create custom
functions, but it's not possible to create custom keywords. That's why
keywords are usually used for "magic" things that would be difficult to
do with just functions.
Also note that if statements check the condition once only, so if we set it to false later the if statement won't notice it.
>>> its_raining = True
>>> if its_raining:
... its_raining = False
... print("It's not raining, but this runs anyway.")
...
It's not raining, but this runs anyway.
>>>
What if we want to print a different message if it's not raining? We could do something like this:
its_raining = True # you can change this to False
its_not_raining = not its_raining # False if its_raining, True otherwise
if its_raining:
print("It's raining!")
if its_not_raining:
print("It's not raining.")
Note that this code example doesn't start with >>>
, so you should
save it to a file and run the file.
Now our program will print a different value depending on what the
value of its_raining
is.
We can also add not its_raining
directly to the second if statement:
its_raining = True
if its_raining:
print("It's raining!")
if not its_raining:
print("It's not raining.")
But we can make it even better by using else
.
its_raining = True
if its_raining:
print("It's raining!")
else:
print("It's not raining.")
The else part simply runs when the if statement doesn't run. It doesn't check the condition again.
>>> its_raining = True
>>> if its_raining:
... its_raining = False
... else:
... print("It's not raining, but this still doesn't run.")
...
>>>
By combining else
with the input function we can make a program that
asks for a password and checks if it's correct.
print("Hello!")
password = input("Enter your password: ")
if password == "secret":
print("That's correct, welcome!")
else:
print("Access denied.")
The program prints different things depending on what we enter:
Hello!
Enter your password: secret
Welcome!
Hello!
Enter your password: lol
Access denied.
Using the input function for passwords doesn't work very well because we can't hide the password with asterisks. There are better ways to get a password from the user, but you shouldn't worry about that just yet.
If we have more than one condition to check, we could do this:
print("Hello!")
word = input("Enter something: ")
if word == "hi":
print("Hi to you too!")
else:
if word == "hello":
print("Hello hello!")
else:
if word == "howdy":
print("Howdyyyy!")
else:
if word == "hey":
print("Hey hey hey!")
else:
if word == "gday m8":
print("Gday 4 u 2!")
else:
print("I don't know what", word, "means.")
This code is a mess. We need to indent more every time we want to check for more words. Here we check for 5 different words, so we have 5 levels of indentation. If we would need to check 30 words, the code would become really wide and it would be hard to work with.
Instead of typing else
, indenting more and typing an if
we can
simply type elif
, which is short for else if
. Like this:
print("Hello!")
word = input("Enter something: ")
if word == "hi":
print("Hi to you too!")
elif word == "hello":
print("Hello hello!")
elif word == "howdy":
print("Howdyyyy!")
elif word == "hey":
print("Hey hey hey!")
elif word == "gday m8":
print("Gday 4 u 2!")
else:
print("I don't know what", word, "means.")
Now the program is shorter and much easier to read.
Note that the elif
parts only run if nothing before them matches, and
the else
runs only when none of the elifs
match. If we would have
used if
instead, all possible values would be always checked and the
else
part would run always except when word is "gday m8"
. This is
why we use elif
instead of if
.
For example, this program prints only hello
...
if 1 == 1:
print("hello")
elif 1 == 2:
print("this is weird")
else:
print("world")
...but this prints hello
and world
:
if 1 == 1:
print("hello")
if 1 == 2:
print("this is weird")
else:
print("world")
Now the else
belongs to the if 1 == 2
part and it has nothing to
do with the if 1 == 1
part. On the other hand, the elif version
grouped the multiple ifs together and the else
belonged to all of
them. Adding a blank line makes this obvious:
if 1 == 1:
print("hello")
if 1 == 2:
print("this is weird")
else:
print("world")
In general, adding blank lines to appropriate places is a good idea. If you are asked to "fix code", feel free to add missing blank lines.
- If a code example starts with
>>>
run it on the interactive prompt. If it doesn't, write it to a file and run that file. - Indentation is important in Python.
- Indented code under an if statement runs if the condition is true.
- We can also add an else statement. Indented code under it will run if the code under the if statement does not run.
- elif is short for else if.
-
This program contains several problems. Copy-paste it to a file, then try to run it, fix the errors you got, try to run it again and keep going until it works.
print(Hello!) something == input('Enter something: ) print('You entered:' something)
-
Fix this program the same way:
print('Hello!') something = input("Enter something: ") if something = 'hello': print("Hello for you too!") elif something = 'hi' print('Hi there!') else: print("I don't know what," something, "means.")
-
Write a program into a file that asks the user to write a word and then prints that word 1000 times. For example, if the user enters
hi
the program would replyhihihihihihihihi
... -
Add spaces between the words, so the output is like
hi hi hi hi
... -
Make something that asks the user to enter two words, and prints 1000 of each with spaces in between. For example, if the user enters
hello
andhi
the program would printhello hi hello hi hello hi hello hi hello hi
... -
Make a program that asks for a password and prints
Welcome!
,Access denied
orYou didn't enter anything
depending on whether the user entered the correct password, a wrong password, or nothing at all by pressing Enter without typing anything.
Python strings are just pieces of text.
>>> our_string = "Hello World!"
>>> our_string
'Hello World!'
>>>
So far we know how to add them together.
>>> "I said: " + our_string
'I said: Hello World!'
>>>
We also know how to repeat them multiple times.
>>> our_string * 3
'Hello World!Hello World!Hello World!'
>>>
Python strings are immutable.
That's just a fancy way to say that
they cannot be changed in-place, and we need to create a new string to
change them. Even some_string += another_string
creates a new string.
Python will treat that as some_string = some_string + another_string
,
so it creates a new string but it puts it back to the same variable.
+
and *
are nice, but what else can we do with strings?
Slicing is really simple. It just means getting a part of the string. For example, to get all characters between the second place between the characters and the fifth place between the characters, we can do this:
>>> our_string[2:5]
'llo'
>>>
So the syntax is like some_string[start:end]
.
What happens if we slice with negative values?
>>> our_string[-5:-2]
'orl'
>>>
It turns out that slicing with negative values simply starts counting from the end of the string.
If we don't specify the beginning it defaults to 0, and if we don't specify the end it defaults to the length of the string. For example, we can get everything except the first or last character like this:
>>> our_string[1:]
'ello World!'
>>> our_string[:-1]
'Hello World'
>>>
Remember that strings can't be changed in-place.
>>> our_string[:5] = 'Howdy'
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'str' object does not support item assignment
>>>
There's also a step argument we can give to our slices, but I'm not going to talk about it now.
So now we know how slicing works. But what happens if we forget the :
?
>>> our_string[1]
'e'
>>>
That's interesting. We got a string that is only one character long. But
the first character of Hello World!
should be H
, not e
, so why did
we get an e?
Programming starts at zero. Indexing strings also starts at zero. The
first character is our_string[0]
, the second character is
our_string[1]
, and so on.
>>> our_string[0]
'H'
>>> our_string[1]
'e'
>>> our_string[2]
'l'
>>> our_string[3]
'l'
>>> our_string[4]
'o'
>>>
How about negative values?
>>> our_string[-1]
'!'
>>>
We got the last character.
But why didn't that start at zero? our_string[-1]
is the last
character, but our_string[1]
is not the first character!
That's because 0 and -0 are equal, so indexing with -0 would do the same thing as indexing with 0.
Indexing with negative values works like this:
Python's strings have many useful methods.
The official documentation
covers them all, but I'm going to just show some of the most commonly
used ones briefly. Python also comes with built-in documentation about
the string methods and we can run help(str)
to read it. We can also
get help about one string method at a time, like help(str.upper)
.
Again, nothing can modify strings in-place. Most string methods
return a new string, but things like our_string = our_string.upper()
still work because the new string is assigned to the old variable.
Also note that all of these methods are used like our_string.stuff()
,
not like stuff(our_string)
. The idea with that is that our string
knows how to do all these things, like our_string.stuff()
, we don't
need a separate function that does these things like stuff(our_string)
.
We'll learn more about methods later.
Here's an example with some of the most commonly used string methods:
>>> our_string.upper()
'HELLO WORLD!'
>>> our_string.lower()
'hello world!'
>>> our_string.startswith('Hello')
True
>>> our_string.endswith('World!')
True
>>> our_string.endswith('world!') # Python is case-sensitive
False
>>> our_string.replace('World', 'there')
'Hello there!'
>>> our_string.replace('o', '@', 1) # only replace one o
'Hell@ World!'
>>> ' hello 123 '.lstrip() # left strip
'hello 123 '
>>> ' hello 123 '.rstrip() # right strip
' hello 123'
>>> ' hello 123 '.strip() # strip from both sides
'hello 123'
>>> ' hello abc'.rstrip('cb') # strip c's and b's from right
' hello a'
>>> our_string.ljust(30, '-')
'Hello World!------------------'
>>> our_string.rjust(30, '-')
'------------------Hello World!'
>>> our_string.center(30, '-')
'---------Hello World!---------'
>>> our_string.count('o') # it contains two o's
2
>>> our_string.index('o') # the first o is our_string[4]
4
>>> our_string.rindex('o') # the last o is our_string[7]
7
>>> '-'.join(['hello', 'world', 'test'])
'hello-world-test'
>>> 'hello-world-test'.split('-')
['hello', 'world', 'test']
>>> our_string.upper()[3:].startswith('LO WOR') # combining multiple things
True
>>>
The things in square brackets that the split method gave us and we gave to the join method were lists. We'll talk more about them later.
To add a string in the middle of another string, we can do something like this:
>>> name = 'Akuli'
>>> 'My name is ' + name + '.'
'My name is Akuli.'
>>>
But that gets complicated if we have many things to add.
>>> channel = '##learnpython'
>>> network = 'freenode'
>>> "My name is " + name + " and I'm on the " + channel + " channel on " + network + "."
"My name is Akuli and I'm on the ##learnpython channel on freenode."
>>>
Instead it's recommended to use string formatting. It means putting other things in the middle of a string.
Python has multiple ways to format strings. One is not necessarily better than others, they are just different. Here's a few ways to solve our problem:
-
.format()
-formatting, also known as new-style formatting. This formatting style has a lot of features, but it's a little bit more typing than%s
-formatting.>>> "Hello {}.".format(name) 'Hello Akuli.' >>> "My name is {} and I'm on the {} channel on {}.".format(name, channel, network) "My name is Akuli and I'm on the ##learnpython channel on freenode." >>>
-
%s
-formatting, also known as old-style formatting. This has less features than.format()
-formatting, but'Hello %s.' % name
is shorter and faster to type than'Hello {}.'.format(name)
. I like to use%s
formatting for simple things and.format
when I need more powerful features.>>> "Hello %s." % name 'Hello Akuli.' >>> "My name is %s and I'm on the %s channel on %s." % (name, channel, network) "My name is Akuli and I'm on the ##learnpython channel on freenode." >>>
In the second example we had
(name, channel, network)
on the right side of the%
sign. It was a tuple, and we'll talk more about them later.If we have a variable that may be a tuple we need to wrap it in another tuple when formatting:
>>> thestuff = (1, 2, 3) >>> "we have %s" % thestuff Traceback (most recent call last): File "<stdin>", line 1, in <module> TypeError: not all arguments converted during string formatting >>> "we have %s and %s" % ("hello", thestuff) 'we have hello and (1, 2, 3)' >>> "we have %s" % (thestuff,) 'we have (1, 2, 3)' >>>
Here
(thestuff,)
was a tuple that contained nothing butthestuff
. -
f-strings are even less typing, but new in Python 3.6. Use this only if you know that nobody will need to run your code on Python versions older than 3.6. Here the f is short for "format", and the content of the string is same as it would be with
.format()
but we can use variables directly.>>> f"My name is {name} and I'm on the {channel} channel on {network}." "My name is Akuli and I'm on the ##learnpython channel on freenode." >>>
All of these formatting styles have many other features also:
>>> 'Three zeros and number one: {:04d}'.format(1)
'Three zeros and number one: 0001'
>>> 'Three zeros and number one: %04d' % 1
'Three zeros and number one: 0001'
>>>
If you need to know more about formatting I recommend reading this.
We can use in
and not in
to check if a string contains another
string.
>>> our_string = "Hello World!"
>>> "Hello" in our_string
True
>>> "Python" in our_string
False
>>> "Python" not in our_string
True
>>>
We can get the length of a string with the len
function. The name
len
is short for "length".
>>> len(our_string) # 12 characters
12
>>> len('') # no characters
0
>>> len('\n') # python thinks of \n as one character
1
>>>
We can convert strings, integers and floats with each other with
str
, int
and float
. They aren't actually functions, but they
behave a lot like functions. We'll learn more about what they really
are later.
>>> str(3.14)
'3.14'
>>> float('3.14')
3.14
>>> str(123)
'123'
>>> int('123')
123
>>>
Giving an invalid string to int
or float
produces an error
message.
>>> int('lol')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: invalid literal for int() with base 10: 'lol'
>>> float('hello')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: could not convert string to float: 'hello'
>>>
-
Slicing returns a copy of a string with indexes from one index to another index.
-
Indexing returns one character of a string. Remember that we don't need a
:
with indexing. -
Python has many string methods. Use the documentation or
help(str)
when you don't rememeber something about them. -
String formatting means adding other things to the middle of a string. There are multiple ways to do this in Python. You should know how to use at least one of these ways.
-
The
in
keyword can be used for checking if a string contains another string. -
len(string)
returns string's length. -
We can use
str
,int
andfloat
to convert values to different types.
-
Fix this program.
print("Hello!") word1 = input("Enter something: ") word2 = input("Enter another thing: ") word3 = input("Enter a third thing: ") word4 = input("And yet another thing: ") print("You entered " + word1 + ", " + word2 + ", " + word3 + " and " + word4 + ".")
-
This program is supposed to say something loudly. Fix it.
message = input("What do you want me to say? ") message.upper print(message, "!!!") print(message, "!!!") print(message, "!!!")
-
Make a program to ask a string from the user and check if it is a palindrome.
(Hint: A string is a palindrome if it is the same when reversed. Google how to reverse a string.)
Sometimes we may end up doing something like this.
name1 = 'wub_wub'
name2 = 'theelous3'
name3 = 'RubyPinch'
name4 = 'go|dfish'
name5 = 'Nitori'
name = input("Enter your name: ")
if name == name1 or name == name2 or name == name3 or name == name4 or name == name5:
print("I know you!")
else:
print("Sorry, I don't know who you are :(")
This code works just fine, but there's a problem. The name check is repetitive, and adding a new name requires adding even more repetitive, boring checks.
Instead of adding a new variable for each name it might be better to store all names in one variable. This means that our one variable needs to point to multiple values. An easy way to do this is using a list:
names = ['wub_wub', 'theelous3', 'Nitori', 'RubyPinch', 'go|dfish']
Here the names
variable points to a list, which then points to
strings.
Let's open the >>>
prompt and create a name list.
>>> names = ['wub_wub', 'theelous3', 'RubyPinch', 'go|dfish', 'Nitori']
>>> names
['wub_wub', 'theelous3', 'RubyPinch', 'go|dfish', 'Nitori']
>>>
There's many things we can do with strings and some of these things also work with lists.
>>> len(names) # len is short for length, we have 5 names
5
>>> names + ['Akuli'] # create a new list with me in it
['wub_wub', 'theelous3', 'RubyPinch', 'go|dfish', 'Nitori', 'Akuli']
>>> ['theelous3', 'RubyPinch'] * 2 # repeating
['theelous3', 'RubyPinch', 'theelous3', 'RubyPinch']
>>>
With strings indexing and slicing both returned a string, but with lists we get a new list when we're slicing and an element from the list if we're indexing.
>>> names[:2] # first two names
['wub_wub', 'theelous3']
>>> names[0] # the first name
'wub_wub'
>>>
If we want to check if the program knows a name all we need to
do is to use the in
keyword.
>>> 'lol' in names
False
>>> 'RubyPinch' in names
True
>>>
We can't use this for checking if a list of names is a part of our name list.
>>> ['RubyPinch', 'go|dfish'] in names
False
>>> ['RubyPinch'] in names
False
>>>
Lists have a few useful
methods.
Some of the most commonly used ones are append, extend and remove.
append
adds an item to the end of a list, extend
adds
multiple items from another list and remove
removes an item.
>>> names
['wub_wub', 'theelous3', 'RubyPinch', 'go|dfish', 'Nitori']
>>> names.remove('theelous3') # sorry theelous3
>>> names.remove('go|dfish') # and sorry go|dfish
>>> names
['wub_wub', 'RubyPinch', 'Nitori']
>>> names.append('Akuli') # let's add me here
>>> names
['wub_wub', 'RubyPinch', 'Nitori', 'Akuli']
>>> names.extend(['go|dfish', 'theelous3']) # wb guys
>>> names
['wub_wub', 'RubyPinch', 'Nitori', 'Akuli', 'go|dfish', 'theelous3']
>>>
Note that remove
removes only the first match it finds.
>>> names = ['theelous3', 'go|dfish', 'theelous3']
>>> names.remove('theelous3')
>>> names # the second theelous3 is still there!
['go|dfish', 'theelous3']
>>>
If we need to remove all matching items we can use a simple while loop. We'll talk more about loops in the next chapter
>>> names = ['theelous3', 'go|dfish', 'theelous3']
>>> while 'theelous3' in names:
... names.remove('theelous3')
...
>>> names
['go|dfish']
>>>
Another useful thing about lists is list comprehension. It's a handy way to construct a list in single line. It often makes code cleaner, shorter and easier to read.
>>> numbers = [1,2,3,4,5]
>>> numbers_squared = [number ** 2 for number in numbers]
>>> numbers_squared
[1, 4, 9, 16, 25]
>>>
Without a list comprehension, doing the same thing looks like this:
>>> numbers = [1,2,3,4,5]
>>> numbers_squared = []
>>> for number in numbers:
... numbers_squared.append(number**2)
>>> numbers_squared
[1, 4, 9, 16, 25]
>>>
We can also use slicing and indexing to change the content:
>>> names = ['theelous3', 'LOL', 'RubyPinch', 'go|dfish', 'Nitori']
>>> names[1] = 'wub_wub' # replace LOL with wub_wub
>>> names
['theelous3', 'wub_wub', 'RubyPinch', 'go|dfish', 'Nitori']
>>>
As you can see, list can be changed in-place. In other words, they are mutable. Integers, floats, strings and many other built-in types can't, so they are immutable.
With strings we did something to them
and then set the result back to the same variable, like
message = message.strip()
. This just doesn't work right with
most mutable things because they're designed to be changed in-place.
>>> names = names.remove('Akuli')
>>> print(names) # now it's None!
None
>>>
This is the same thing that happened way back when we assigned print's return value to a variable
After working with lists a while you'll find out that they behave like this:
>>> a = [1, 2, 3]
>>> b = a
>>> b.append(4)
>>> a # this changed also!
[1, 2, 3, 4]
>>>
This can be confusing at first, but it's actually easy to
explain. The problem with this code example is the b = a
line.
This is when the is
keyword comes in. It can be used to
check if two variables point to the same thing.
>>> a is b
True
>>>
Typing []
creates a new list every time.
>>> [] is []
False
>>> [1, 2, 3] is [1, 2, 3]
False
>>>
If we need a new list with similar content we can use the
copy
method.
>>> a = [1, 2, 3]
>>> b = a.copy()
>>> b is a
False
>>> b.append(4)
>>> b
[1, 2, 3, 4]
>>> a
[1, 2, 3]
>>>
Tuples are a lot like lists, but they're immutable so they
can't be changed in-place. We create them like lists, but
with ()
instead of []
.
>>> thing = (1, 2, 3)
>>> thing
(1, 2, 3)
>>> thing = ()
>>> thing
()
>>>
If we need to create a tuple that contains only one item we
need to use (item,)
instead of (item)
because (item)
is
used in places like (1 + 2) * 3
.
>>> (3)
3
>>> (3,)
(3,)
>>> (1 + 2) * 3
9
>>> (1 + 2,) * 3
(3, 3, 3)
>>>
It's also possible to create tuples by just separating things with commas and adding no parentheses. Personally I don't like this feature, but some people like to do it this way.
>>> 1, 2, 3
(1, 2, 3)
>>> 'hello',
('hello',)
>>>
Tuples don't have methods like append, extend and remove because they can't change themselves in-place.
>>> stuff = (1, 2, 3)
>>> stuff.append(4)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'tuple' object has no attribute 'append'
>>>
So, why the heck would we use tuples instead of lists? There are some cases when we don't want mutability, but there are also cases when Python programmers just like to use tuples. If you want to know more about this you can read Ned Batchelder's blog post.
- Lists are a way to store multiple values in one variable.
- Lists can be changed in-place and they have methods that change them in-place, like append, extend and remove.
- Slicing lists returns a new list, and indexing them returns an item from them.
thing = another_thing
does not create a copy ofanother_thing
.- Tuples are like lists, but they can't be changed in-place. They're also used in different places.
Here's the same program we had in the beginning of this tutorial, but using a list:
namelist = ['wub_wub', 'theelous3', 'RubyPinch', 'go|dfish', 'Nitori']
name = input("Enter your name: ")
if name in namelist:
print("I know you!")
else:
print("Sorry, I don't know who you are :(")
-
Fix this program:
namelist = ('wub_wub', 'RubyPinch', 'go|dfish', 'Nitori') namelist.append('pb122') if 'pb122' in namelist: print("Now I know pb122!")
-
Fix this program.
print("Hello!") name = input("Enter your name: "), print("Your name is " + name + ".")
-
Fix this program.
namelist = ['wub_wub', 'RubyPinch', 'go|dfish', 'Nitori'] namelist = namelist.extend('theelous3') if input("Enter your name: ") in namelist: print("I know you!") else: print("I don't know you.")
In programming, a loop means repeating something multiple times. There are different kinds of loops:
- While loops repeat something while a condition is true.
- Until loops repeat something while a condition is false.
- For loops repeat something for each element of something.
We'll talk about all of these in this tutorial.
Now we know how if statements work.
its_raining = True
if its_raining:
print("Oh crap, it's raining!")
While loops are really similar to if statements.
its_raining = True
while its_raining:
print("Oh crap, it's raining!")
# we'll jump back to the line with the word "while" from here
print("It's not raining anymore.")
If you're not familiar with while loops, the program's output may be a bit surprising:
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
Oh crap, it's raining!
(much more raining)
Again, this program does not break your computer. It just prints the same thing multiple times. We can interrupt it by pressing Ctrl+C.
In this example, its_raining
was the condition. If something in
the while loop would have set its_raining
to False, the loop would
have ended and the program would have printed It's not raining anymore
.
Let's actually create a program that does just that:
its_raining = True
while its_raining:
print("It's raining!")
answer = input("Or is it? (y=yes, n=no) ")
if answer == 'y':
print("Oh well...")
elif answer == 'n':
its_raining = False # end the while loop
else:
print("Enter y or n next time.")
print("It's not raining anymore.")
Running the program may look like this:
It's raining!
Or is it? (y=yes, n=no) i dunno
Enter y or n next time.
It's raining!
Or is it? (y=yes, n=no) y
Oh well...
It's raining!
Or is it? (y=yes, n=no) n
It's not raining anymore.
The while loop doesn't check the condition all the time, it only checks it in the beginning.
>>> its_raining = True
>>> while its_raining:
... its_raining = False
... print("It's not raining, but the while loop doesn't know it yet.")
...
It's not raining, but the while loop doesn't know it yet.
>>>
We can also interrupt a loop even if the condition is still true using
the break
keyword. In this case, we'll set condition to True and rely
on nothing but break
to end the loop.
while True:
answer = input("Is it raining? (y=yes, n=no) ")
if answer == 'y':
print("It's raining!")
elif answer == 'n':
print("It's not raining anymore.")
break # end the loop
else:
print("Enter y or n.")
The program works like this:
Is it raining? (y=yes, n=no) who knows
Enter y or n.
Is it raining? (y=yes, n=no) y
It's raining!
Is it raining? (y=yes, n=no) n
It's not raining anymore.
Unlike setting the condition to False, breaking the loop ends it immediately.
>>> while True:
... break
... print("This is never printed.")
...
>>>
Python doesn't have until loops. If we need an until loop, we can use
while not
:
raining = False
while not raining:
print("It's not raining.")
if input("Is it raining? (y/n) ") == 'y':
raining = True
print("It's raining!")
Let's say we have a list of things we want to print. To print each item in it, we could just do a bunch of prints:
stuff = ['hello', 'hi', 'how are you doing', 'im fine', 'how about you']
print(stuff[0])
print(stuff[1])
print(stuff[2])
print(stuff[3])
print(stuff[4])
The output of the program is like this:
hello
hi
how are you doing
im fine
how about you
But this is only going to print five items, so if we add something to stuff, it's not going to be printed. Or if we remove something from stuff, we'll get an error saying "list index out of range".
We could also create an index variable, and use a while loop:
>>> stuff = ['hello', 'hi', 'how are you doing', 'im fine', 'how about you']
>>> length_of_stuff = len(stuff)
>>> index = 0
>>> while index < length_of_stuff:
... print(stuff[index])
... index += 1
...
hello
hi
how are you doing
im fine
how about you
>>>
But we have len()
and an index variable we need to increment and a
while loop and many other things to worry about. That's a lot of work
just for printing each item.
This is when for loops come in:
>>> for thing in stuff:
... # this is repeated for each element of stuff, that is, first
... # for stuff[0], then for stuff[1], etc.
... print(thing)
...
hello
hi
how are you doing
im fine
how about you
>>>
Without the comments, that's only two simple lines, and one variable. Much better than anything else we tried before.
>>> for thing in stuff:
... print(thing)
...
hello
hi
how are you doing
im fine
how about you
>>>
Note that for thing in stuff:
is not same as for (thing in stuff):
.
Here the in
keyword is just a part of the for loop and it has a
different meaning than it would have if we had thing in stuff
without
a for
. Trying to do for (thing in stuff):
creates an error.
Right now the while loop version might seem easier to understand for you, but later you'll realize that for loops are much easier to work with than while loops and index variables, especially in large projects. For looping is also a little bit faster than while looping with an index variable.
For loops are not actually limited to lists. We can for loop over many other things also, including strings and tuples. For looping over a tuple gives us its items, and for looping over a string gives us its characters as strings of length one.
>>> for short_string in 'abc':
... print(short_string)
...
a
b
c
>>> for item in (1, 2, 3):
... print(item)
...
1
2
3
>>>
If we can for loop over something, then that something is iterable. Lists, tuples and strings are all iterable.
There's only one big limitation with for looping over lists. We shouldn't modify the list in the for loop. If we do, the results can be surprising:
>>> stuff = ['hello', 'hi', 'how are you doing', 'im fine', 'how about you']
>>> for thing in stuff:
... stuff.remove(thing)
...
>>> stuff
['hi', 'im fine']
>>>
Instead, we can create a copy of stuff and loop over it.
>>> stuff = ['hello', 'hi', 'how are you doing', 'im fine', 'how about you']
>>> for thing in stuff.copy():
... stuff.remove(thing)
...
>>> stuff
[]
>>>
Or if we just want to clear a list, we can use the clear
list method:
>>> stuff = ['hello', 'hi', 'how are you doing', 'im fine', 'how about you']
>>> stuff.clear()
>>> stuff
[]
>>>
- A loop means repeating something multiple times.
- While loops repeat something as long as a condition is true, and they check the condition only in the beginning.
- For loops can be used for repeating something to each item in a list.
- An iterable is something that can be for looped over.
- The
break
keyword can be used to interrupt the innermost loop at any time.
Repeat something an endless amount of times.
message = input("What do you want me to say? ")
while True:
print(message, "!!!")
Ask the user to enter five things and print them.
things = []
print("Enter 5 things: ")
while len(things) < 5:
thing = input("> ")
things.append(thing)
print("You entered these things:")
for thing in things:
print(thing)
Ask the user a bunch of questions.
questions_and_answers = [
# [question, answer], ...
["What is 2+4? ", "6"],
["What is 2-4? ", "-2"],
["What is 2*4? ", "8"],
["What is 2/4? ", "0.5"],
# You could add more questions, but the code that asks them
# wouldn't need to be changed in any way.
]
for qa in questions_and_answers:
while True:
if input(qa[0]) == qa[1]:
print("Correct!")
break
else:
print("That's not what I was thinking of... Try again.")
Store a list of names and let the user check if the program knows the user.
# You can add names here so the program will know them automatically
# when it starts.
namelist = []
print("Options:")
print(" 0 Quit")
print(" 1 Check if I know you")
print(" 2 Introduce yourself to me")
print(" 3 Make me forget you")
print(" 4 Print a list of people I know")
print() # print an empty line
while True:
option = input("Choose an option: ")
# Things like option == 0 don't work because option is a string
# and it needs to be compared with a string:
# >>> 0 == 0
# True
# >>> '0' == '0'
# True
# >>> 0 == '0'
# False
if option == '0':
print("Bye!")
break
elif option == '1':
name = input("Enter your name: ")
if name in namelist:
print("I know you! :D")
else:
print("I don't know you :/")
elif option == '2':
name = input("Enter your name: ")
if name in namelist:
print("I knew you already.")
else:
namelist.append(name)
print("Now I know you!")
elif option == '3':
name = input("Enter your name: ")
if name in namelist:
namelist.remove(name)
print("Now I don't know you.")
else:
print("I didn't know you to begin with.")
elif option == '4':
if namelist == []:
print("I don't know anybody yet.")
else:
for name in namelist:
print("I know %s!" % name)
else:
print("I don't understand :(")
print()
-
This code is supposed to print the numbers 1,2,3,4,5. Fix it.
things = str([1, 2, 3, 4, 5]) for thing in things: print(thing)
-
This code is supposed to print
[1, 2, 3, 4, 5, 6]
. Fix it without changing thebefore
list.before = [[1, 2], [3, 4], [5, 6]] after = [] for number in before: after.append(number) print(after)
-
This program is supposed to convert everything in a list to integers and then calculate their sum. It should print 6 because
1 + 2 + 3
is 6. Fix the program.input = ['1', '2', '3'] for string in input: numbers = [] numbers.append(int(string)) result = 0 for n in numbers: result + n print("their sum is", result)
-
This program is supposed to print
[1, 2, 3]
. Fix it.numbers = ['1', '2', '3'] for number in numbers: number = int(number) print(numbers)
-
Make a program that prints a pyramid like shown below. Ask the user to type the number of rows needed.
OUTPUT for 5 rows 1 1 2 1 2 3 1 2 3 4 1 2 3 4 5
-
Make a program to get a pyramid like shown below where user can type the number of rows needed.
OUTPUT for 5 rows 1 2 3 4 5 2 3 4 5 3 4 5 4 5 5
Now we know how lists and tuples work and how to for loop over them. If we make some kind of program that needs to keep track of people's names and favorite pets, we can use a list for that:
names_and_pets = [
('horusr', 'cats'),
('caisa64', 'cats and dogs'),
('__Myst__', 'cats'),
]
Then to check if cats are horusr's favorite pets we can do
('horusr', 'cats') in names_and_pets
. Or we can add new people's
favorite pets easily by appending new (name, pets)
tuples to the list.
But what if we need to check if we know anything about someone's
favorite pets? 'caisa64' in names_and_pets
is always False because the
pet list consists of (name, pets)
pairs instead of just names, so we
need to for loop over the whole pet list:
found_caisa64 = False
for pair in names_and_pets:
if pair[0] == 'caisa64':
found_caisa64 = True
break
if found_caisa64:
# do something
Or what if we need to find out what caisa64's favorite pets are? That also requires going through the whole list.
pets = None
for pair in names_and_pets:
if pair[0] == 'caisa64':
pets = pair[1]
break
# make sure pets is not None and do something with it
As you can see, a list of (name, pets)
pairs is not an ideal
way to store names and favorite pets.
A better way to store information about favorite pets might be a dictionary:
favorite_pets = {
'horusr': 'cats',
'caisa64': 'cats and dogs',
'__Myst__': 'cats',
}
Here 'horusr'
and 'caisa64'
are keys in the dictionary, and
'cats'
and 'cats and dogs'
are their values. Dictionaries are
often named by their values. This dictionary has favorite pets as its
values so I named the variable favorite_pets
.
There are a few big differences between dictionaries and lists of pairs:
- Dictionaries are not ordered. There are no guarantees about which
order the
name: pets
pairs appear in when we do something with the dictionary. - Checking if a key is in the dictionary is simple and fast. We don't need to for loop through the whole dictionary.
- Getting the value of a key is also simple and fast.
- We can't have the same key in the dictionary multiple times, but multiple different keys can have the same value. This means that multiple people can't have the same name, but they can have the same favorite pets.
But wait... this is a lot like variables are! Our variables are not ordered, getting a value of a variable is fast and easy and we can't have multiple variables with the same name.
Variables are actually stored in a dictionary. We can get that dictionary with the globals function. In this dictionary, keys are variable names and values are what our variables point to.
>>> globals()
{'names_and_pets': [('horusr', 'cats'),
('caisa64', 'cats and dogs'),
('__Myst__', 'cats')],
'favorite_pets': {'__Myst__': 'cats',
'caisa64': 'cats and dogs',
'horusr': 'cats'},
...many other things we don't need to care about...
}
>>>
So if you have trouble remembering how dictionaries work just compare them to variables. A dictionary is a perfect way to store these names and favorite pets. We don't care about which order the names and pets were added in, it's impossible to add the same name multiple times and getting someone's favorite pets is easy.
Dictionaries have some similarities with lists. For example, both lists and dictionaries have a length.
>>> len(names_and_pets) # contains three elements
3
>>> len(favorite_pets) # contains three key:value pairs
3
>>>
We can get a value of a key with the_dict[key]
. This is a lot easier
and faster than for-looping over a list of pairs.
>>> favorite_pets['caisa64']
'cats and dogs'
>>> favorite_pets['__Myst__']
'cats'
>>>
Trying to get the value of a non-existing key gives us an error.
>>> favorite_pets['Akuli']
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
KeyError: 'Akuli'
>>>
But we can add new key: value
pairs or change the values of existing
keys by doing the_dict[key] = value
.
>>> favorite_pets['Akuli'] = 'penguins'
>>> favorite_pets['Akuli']
'penguins'
>>> favorite_pets['Akuli'] = 'dogs'
>>> favorite_pets['Akuli']
'dogs'
>>> favorite_pets
{'__Myst__': 'cats',
'Akuli': 'dogs',
'horusr': 'cats',
'caisa64': 'cats and dogs'}
>>>
For looping over a dictionary gets its keys, and checking if something is in the dictionary checks if the dictionary has a key like that. This can be confusing at first but you'll get used to this.
>>> 'Akuli' in favorite_pets
True
>>> 'dogs' in favorite_pets
False
>>> for name in favorite_pets:
... print(name)
...
caisa64
Akuli
__Myst__
horusr
>>>
Dictionaries have a values method that we can use if we want to do something with the values:
>>> favorite_pets.values()
dict_values(['dogs', 'cats', 'cats and dogs', 'cats'])
>>>
The values method returned a dict_values
object. Things like this
behave a lot like lists and usually we don't need to convert them to
lists.
>>> for pets in favorite_pets.values():
... print(pets)
...
dogs
cats
cats and dogs
cats
>>>
We can do things like list(favorite_pets.values())
if we need a real
list for some reason, but doing that can slow down our program if the
dictionary is big. There's also a keys method, but usually we don't need
it because the dictionary itself behaves a lot like a list of keys.
If we need both keys and values we can use the items method with the
for first, second in thing
trick.
>>> favorite_pets.items()
dict_items([('Akuli', 'dogs'),
('__Myst__', 'cats'),
('caisa64', 'cats and dogs'),
('horusr', 'cats')])
>>> for name, pets in favorite_pets.items():
... print("{} are {}'s favorite pets".format(pets, name))
...
dogs are Akuli's favorite pets
cats are __Myst__'s favorite pets
cats and dogs are caisa64's favorite pets
cats are horusr's favorite pets
>>>
This is also useful for checking if the dictionary has a key: value
pair.
>>> ('horusr', 'cats') in favorite_pets.items()
True
>>> ('horusr', 'dogs') in favorite_pets.items()
False
>>>
Sometimes it might be handy to use lists as dictionary keys, but it just doesn't work. I'm not going to explain why Python doesn't allow this because usually we don't need to worry about that.
>>> stuff = {['a', 'b']: 'c', ['d', 'e']: 'f'}
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: unhashable type: 'list'
>>>
On the other hand, tuples work just fine:
>>> stuff = {('a', 'b'): 'c', ('d', 'e'): 'f'}
>>> stuff
{('a', 'b'): 'c', ('d', 'e'): 'f'}
>>>
The values of a dictionary can be anything.
>>> stuff = {'a': [1, 2, 3], 'b': [4, 5, 6]}
>>> stuff
{'a': [1, 2, 3], 'b': [4, 5, 6]}
>>>
- Dictionaries consist of
key: value
pairs. - Variables are stored in a dictionary with their names as keys, so
dictionaries behave a lot like variables:
- Dictionaries are not ordered.
- Setting or getting the value of a key is simple and fast.
- Dictionaries can't contain the same key more than once.
- For-looping over a dictionary loops over its keys, and checking if
something is in the dictionary checks if the dictionary has a key
like that. The
values()
anditems()
methods return things that behave like lists of values or(key, value)
pairs instead.
This program counts how many times words appear in a sentence.
sentence.split()
creates a list of words in the sentence, see
help(str.split)
for more info.
sentence = input("Enter a sentence: ")
counts = {} # {word: count, ...}
for word in sentence.split():
if word in counts:
# we have seen this word before
counts[word] += 1
else:
# this is the first time this word occurs
counts[word] = 1
print() # display an empty line
for word, count in counts.items():
if count == 1:
# "1 times" looks weird
print(word, "appears once in the sentence")
else:
print(word, "appears", count, "times in the sentence")
Running the program might look like this:
Enter a sentence: this is a test and this is quite long because this is a test
is appears 3 times in the sentence
long appears once in the sentence
a appears 2 times in the sentence
because appears once in the sentence
this appears 3 times in the sentence
quite appears once in the sentence
and appears once in the sentence
test appears 2 times in the sentence
TODO: Exercises.
It's probably been a while since you read about using functions. Read about it again if you need to.
Have a look at this code:
print("************")
print("Hello World!")
print("************")
print("*************")
print("Enter a word:")
print("*************")
word = input()
if word == 'python':
print("*******************")
print("You entered Python!")
print("*******************")
else:
print("**************************")
print("You didn't enter Python :(")
print("**************************")
Then compare it to this code:
print_box("Hello World!")
print_box("Enter a word:")
word = input()
if word == 'python':
print_box("You entered Python!")
else:
print_box("You didn't enter Python :(")
In this tutorial we'll learn to define a print_box
function
that prints text in a box. We can write the code for printing the
box once, and then use it multiple times anywhere in the program.
Dividing a long program into simple functions also makes the code easier to work with. If there's a problem with the code we can test the functions one by one and find the problem easily.
The pass
keyword does nothing.
>>> pass
>>>
Let's use it to define a function that does nothing.
>>> def do_nothing():
... pass
...
>>> do_nothing
<function do_nothing at 0x7f56b74e9598>
>>>
Seems to be working so far, we have a function. It's just a value that
is assigned to a variable called do_nothing
. You can ignore the
0xblablabla
stuff for now.
The pass
is needed here because without it, Python doesn't know when
the function ends and it gives us a syntax error. We don't need the
pass
when our functions contain something else.
Let's see what happens if we call our function.
>>> do_nothing()
>>>
There we go. It did nothing at all.
Maybe we could just do something in the function instead?
>>> def print_hi():
... print("Hi!")
...
>>> print_hi()
Hi!
>>>
It's working. How about printing a variable in the function?
>>> def print_message():
... print(message)
...
>>> message = "Hello World!"
>>> print_message()
Hello World!
>>>
Again, it works. How about setting a variable in the function?
>>> def get_username():
... username = input("Username: ")
...
>>> get_username()
Username: me
>>> username
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'username' is not defined
>>>
That was weird! Why didn't that work?
So far we have used nothing but global variables. They are called globals because the same variables are available anywhere in our program, even in functions.
>>> a = 1
>>> b = "hi"
>>> c = "hello"
>>> def print_abc():
... print(a, b, c)
...
>>> print_abc()
1 hi hello
>>>
But there are also local variables. They exist only inside functions, and they are deleted when the function exits.
>>> def thingy():
... d = "hello again, i'm a local variable"
... print('inside thingy:', d)
...
>>> thingy()
inside thingy: hello again, i'm a local variable
>>> d
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'd' is not defined
>>>
However, modifying a global variable in-place from a function is easy.
>>> stuff = ['global stuff']
>>> def add_stuff():
... stuff.append('local stuff')
...
>>> add_stuff()
>>> stuff
['global stuff', 'local stuff']
>>>
This only works for changing in-place, we cannot assign a new value to the variable.
>>> def set_stuff_to_something_new():
... stuff = ['more local stuff']
...
>>> set_stuff_to_something_new()
>>> stuff
['global stuff', 'local stuff']
>>>
Note: This section has nothing to do with the input
function that
is used like word = input("enter something: ")
.
So far our functions seem to be really isolated from the rest of our code, and it sucks! But they really are not as isolated as you might think they are.
Let's think about what the print function does. It takes an argument and prints it. Maybe a custom function could also take an argument?
>>> def print_twice(message):
... print(message)
... print(message)
...
>>>
Here message
is an argument. When we call the function we'll get a
local variable called message that will point to whatever we passed
to print_twice
.
This function can be called in two ways:
-
Using a positional argument.
This is the recommended way for functions that take only one or two arguments. I would do this in my code.
>>> print_twice("hi") hi hi >>>
When the function was running it had a local
message
variable that pointed to"hi"
. The function printed it twice.Positional arguments are great for simple things, but if our function takes many positional arguments it may be hard to tell which argument is which.
-
Using a keyword argument:
>>> print_twice(message="hi") hi hi >>>
The name "keyword argument" is a little bit confusing because keyword arguments don't actually have anything to do with keywords (
if
,else
etc). Keyword arguments are just a way to give names for our arguments.Keyword arguments are great when our function needs to take many arguments, because each argument has a name and it's easy to see which argument is which.
Also note that there are no spaces around the
=
sign. This is just a small style detail that Python programmers like to do becausemessage = "hi"
andsome_function(message="hi")
do two completely different things.
Now it's time to solve our box printing problem:
def print_box(message):
print('*' * len(message))
print(message)
print('*' * len(message))
What if we want to print different characters instead of always printing stars?
We could change our print_box
function to take two arguments:
def print_box(message, character):
print(character * len(message))
print(message)
print(character * len(message))
Then we could change our code to always call print_box
with a star as
the second argument:
print_box("Hello World", "*")
...
But we don't need to change our existing code. We can make the second argument optional by giving it a default value.
def print_box(message, character='*'):
print(character * len(message))
print(message)
print(character * len(message))
We can print a row of stars using the function without specifying a different character in two ways:
-
Using a positional argument.
print_box("Hello World!")
-
Using a keyword argument.
print_box(message="Hello World!")
Or we can give it a different character in a few different ways if we need to:
-
Using two positional arguments.
print_box("Enter a word:", "?")
-
Using two keyword arguments.
print_box(message="Enter a word:", character="?") print_box(character="?", message="Enter a word:")
-
Using one positional argument and one keyword argument.
I would probably do this. If an argument has a default value, I like to use a keyword argument to change it if needed.
print_box("Enter a word:", character="?")
However, this doesn't work:
print_box(character="?", "Enter a word:")
The problem is that we have a keyword argument before a positional argument. Python doesn't allow this. We don't need to worry about this, because if we accidentally call a function like this we will get an error message.
The built-in input function returns a value. Can our function return a value too?
>>> def times_two(thing):
... return thing * 2
...
>>> times_two(3)
6
>>> times_two(5)
10
>>>
Yes, it can. Now typing times_two(3)
to the prompt does the same
thing as typing 6
to the prompt.
We can call the times_two
function and use the result however we
want, just like we can use built-in functions:
>>> times_two(2) + times_two(3) # calculate 4 + 6
10
>>> print('2 * 5 is', times_two(5))
2 * 5 is 10
>>>
Note that returning from a function ends it immediately.
>>> def return_before_print():
... return None
... print("This never gets printed.")
...
>>> return_before_print()
>>>
If we don't have any return statements or we have a return statement that doesn't specify what to return, our function will return None.
>>> def return_none_1():
... pass
...
>>> def return_none_2():
... return
...
>>> print(return_none_1())
None
>>> print(return_none_2())
None
>>>
There are two ways to output information from functions. They can print something or they can return something. So, should we print or return?
Most of the time returning makes functions much easier to use. Think
about the input()
function. It asks the user to enter something, and
then the user enters something and that value is returned. If the input
function would print the value instead of returning it, things like
name = input("Name: ")
wouldn't work and assigning the result to a
variable would be much more difficult. Printing things is fine when we
know that we'll only need to print the result and we'll never need to
assign it to a variable.
If our function returns a value we can always print it, like this:
>>> def return_hi():
... return "hi"
...
>>> print(return_hi())
hi
>>>
Functions are easy to understand, but you need to pay attention to how
you're calling them. Note that some_function
and some_function()
do
two completely different things.
>>> def say_hi():
... print("howdy hi")
...
>>> say_hi # just checking what it is, doesn't run anything
<function say_hi at 0x7f997d2a8510>
>>> say_hi() # this runs it
howdy hi
>>>
Typing say_hi
just gives us the value of the say_hi
variable, which
is the function we defined. But say_hi()
calls that function, so
it runs and gives us a return value. The return value is None so the
>>>
prompt doesn't show it.
But we know that the print function shows None, so what happens if we
wrap the whole thing in print()
?
>>> print(say_hi) # prints the function, just like plain say_hi
<function say_hi at 0x7fd913f58488>
>>> print(say_hi()) # runs the function and then prints the return value
howdy hi
None
>>>
The print(say_hi())
thing looks a bit weird at first, but it's easy to
understand. There's a print inside say_hi
and there's also the print
we just wrote, so two things are printed. Python first ran say_hi()
,
and it returned None so Python did print(None)
. Adding an extra
print()
around a function call is actually a common mistake, and I
have helped many people with this problem.
Ask yes/no questions.
def ask_yes_no(prompt):
while True:
answer = input(prompt + ' (y or n) ')
if answer == 'y' or answer == 'Y':
return True # returning ends the function
if answer == 'n' or answer == 'N':
return False
print("Answer 'y' or 'n'.")
if ask_yes_no("Do you like ice cream?"):
print("You like ice cream!")
else:
print("You don't like ice cream.")
Ask questions with multiple answers.
def ask_until_correct(prompt, correct_options,
error_message="I don't know what you meant."):
while True:
answer = input(prompt + ' ')
if answer in correct_options:
return answer
print(error_message)
colors = ['red', 'yellow', 'blue', 'green', 'orange', 'pink', 'black',
'gray', 'white', 'brown']
choice = ask_until_correct("What's your favorite color?", colors,
error_message="I don't know that color.")
print("Your favorite color is %s!" % choice)
- Functions are a way to write code once, and then use that same code in multiple places.
- Variables inside functions are locals, and variables outside functions are globals. Functions can access all variables, but by default, they can only create and change the value of local variables.
- Functions can take arguments and they can behave differently depending on what arguments they get. Arguments are just local variables.
- Functions can also return one value, like the built-in input function does. Returning also ends the function immediately.
- Return a value instead of printing it if you need to do something with it after calling the function.
- Remember that
thing
,thing()
,print(thing)
andprint(thing())
do different things.
There are many things to learn about functions, and I don't expect you to learn everything at once. However, there are also many free exercises about defining functions you can do.
-
What's wrong with this code?
def ask_name(): name = input("Enter your name: ") ask_name() print("Your name is", name)
-
How about this code?
def get_greeting(): return "Hello World!" print(get_greeting)
-
Why does this print None after greeting the world?
def greet(target): print("Hello", target) print(greet("World"))
-
Find more exercises about defining functions online.
Let's say we want to generate a random number between 1 and 3. The random module is a really easy way to do this:
>>> import random
>>> random.randint(1, 3)
3
>>> random.randint(1, 3)
1
>>> random.randint(1, 3)
3
>>> random.randint(1, 3)
2
>>> random.randint(1, 3)
2
>>>
That's cool... but how does that work?
The first line in the example, import random
, was an
import statement. But what is that random thing that it
gave us?
>>> random
<module 'random' from '/usr/lib/python3.7/random.py'>
>>>
So it's a module, and it comes from a path... but what does all that mean?
Now open the folder that contains your random.py
. On my
system it's /usr/lib/python3.7
, but yours will probably be
different. To open a folder in your file manager you can press
Windows-R on Windows or Alt+F2 on most Linux distributions,
and just type your path there. I don't have an up-to-date copy
of OSX so unfortunately I have no idea what you need to do on
OSX.
You'll see a bunch of files and a few directories in the folder that opens.
All of these .py
files can be imported like we just imported
random.py
. In random.py, there's a line like randint = something
,
so we can use its randint variable with random.randint
after
importing it.
You're probably wondering how a computer can generate random numbers. The random module does different things on different operating systems, but on most systems it reads random noise that several programs on the computer produce and creates random numbers based on that.
Create a random.py
file with the following content:
import random
print("A random number between 1 and 3:", random.randint(1, 3))
Now run the program.
Traceback (most recent call last):
File "random.py", line 1, in <module>
import random
File "/home/akuli/random.py", line 4, in <module>
print("A random number between 1 and 3:", random.randint(1, 3))
AttributeError: 'module' object has no attribute 'randint'
But what was that? Why didn't it work?
TODO: update the -i
instructions.
Let's go ahead and check what's wrong. If you don't use IDLE, you'll
need to pass the -i
option to Python, so if you would normally run
python3 random.py
you should now do python3 -i random.py
. This will
run the file and then give you a >>>
prompt that we can use to check
what's wrong. If you use IDLE, just run the file normally.
We should end up with the same error message, and then a >>>
.
Like this:
Traceback (most recent call last):
File "random.py", line 1, in <module>
import random
File "/home/akuli/random.py", line 4, in <module>
print("A random number between 1 and 3:", random.randint(1, 3))
AttributeError: 'module' object has no attribute 'randint'
>>>
So first of all, what is that random
variable?
>>> random
<module 'random' from '/home/akuli/random.py'>
>>>
What the heck? It's a module called random... but it's not the
random.py
we thought it was. Our random.py
has imported
itself!
So let's go ahead and rename our file from random.py
to
something like ourrandom.py
and try again:
A random number between 1 and 3: 3
There we go, now we don't have our own random.py
so it works.
So seems like that modules can be imported from the directory that
our Python file is in, and also from the directory that the real
random.py
is in. But where else can they come from?
There's a module called sys that contains various things built
into Python. Actually the whole module is built-in, so there's no
sys.py
anywhere. The sys module has a list that contains all
places that modules are searched from:
>>> import sys
>>> sys
<module 'sys' (built-in)>
>>> sys.path
['',
'/usr/lib/python37.zip',
'/usr/lib/python3.7',
'/usr/lib/python3.7/lib-dynload',
'/home/akuli/.local/lib/python3.7/site-packages',
'/usr/local/lib/python3.7/dist-packages',
'/usr/lib/python3/dist-packages']
>>>
So that's where my Python finds its modules. The first thing in my sys.path is an empty string, and in this case it means the current working directory.
Let's create a file called hello.py
that contains a classic greeting:
print("Hello World!")
Let's go ahead and import it, and see how it works.
>>> import hello
Hello World!
>>>
Works as expected, but what happens if we try to import it again?
>>> import hello
>>>
Nothing happened at all.
The reason why the module wasn't loaded twice is simple. In a large project with many files it's normal to import the same module in many files, so it gets imported multiple times. If Python would reload the module every time it's imported, dividing code to multiple files would make the code run slower.
If we need to load the module again we can just exit out of Python and launch it again.
The standard library consists of modules that Python comes with. Here's a very brief overview of what it can do. All of these modules can also do other things, and you can read more about that in the official documentation.
The official documentation is here.
>>> import random
>>> random.randint(1, 3) # 1, 2 or 3
3
>>> colors = ['red', 'blue', 'yellow']
>>> random.choice(colors) # choose one color
'red'
>>> random.sample(colors, 2) # choose two different colors
['yellow', 'red']
>>> random.shuffle(colors) # mix the color list in-place
>>> colors
['yellow', 'red', 'blue']
>>>
The module name "sys" is short for "system", and it contains things that are built into Python. The official documentation is here.
sys.stdin
, sys.stdout
and sys.stderr
are file objects,
just like the file objects that open()
gives us.
>>> import sys
>>> print("Hello!", file=sys.stdout) # this is where prints go by default
Hello!
>>> print("Hello!", file=sys.stderr) # use this for error messages
Hello!
>>> line = sys.stdin.readline() # i will type hello and press enter
hello
>>> line
'hello\n'
>>>
>>> # information about Python's version, behaves like a tuple
>>> sys.version_info
sys.version_info(major=3, minor=7, micro=3, releaselevel='final', serial=0)
>>> sys.version_info[:3] # this is Python 3.7.3
(3, 7, 3)
>>>
>>> sys.exit() # exit out of Python
TODO: why stderr instead of stdout, when to use sys.stdin.readline()
instead of input()
sys.exit()
does the same thing as sys.exit(0)
. The zero means that
the program succeeded, and everything's fine. If our program has an
error we should print an error message to sys.stderr
and then call
sys.exit(1)
. Like this:
if something_went_wrong:
# of course, we need to make real error messages more
# informative than this example is
print("Oh crap! Something went wrong.", file=sys.stderr)
sys.exit(1)
There's no math.py anywhere, math is a built-in module like sys. The official documentation is here.
>>> import math
>>> math
<module 'math' (built-in)>
>>> math.pi # approximate value of Ο
3.141592653589793
>>> math.sqrt(2) # square root of 2
1.4142135623730951
>>> math.radians(180) # convert degrees to radians
3.141592653589793
>>> math.degrees(math.pi/2) # convert radians to degrees
90.0
>>> math.sin(math.pi/2) # sin of 90 degrees or 1/2 Ο radians
1.0
>>>
The official documentation for the time module is here.
>>> import time
>>> time.sleep(1) # wait one second
>>> time.time() # return time in seconds since beginning of the year 1970
1474896325.2394648
>>> time.strftime('%d.%m.%Y %H:%M:%S') # format current time nicely
'07.04.2017 19:08:33'
>>>
You are probably wondering how time.time()
can be used and why its
timing starts from the beginning of 1970. time.time()
is useful for
measuring time differences because we can save its return value to a
variable before doing something, and then afterwards check how much it
changed. There's an example that does this in the example
section.
If you want to know why it starts from 1970 you can read something like
this.
See help(time.strftime)
if you want to know about more format
specifiers like %d
, %m
etc. that time.strftime
can take.
The module name "os" is short for "operating system", and it contains handy functions for interacting with the operating system that Python is running on. The official documentation is here.
>>> import os
>>> os.getcwd() # short for "get current working directory"
'/home/akuli'
>>> os.mkdir('stuff') # create a folder, short for "make directory"
>>>
>>> os.path.isfile('hello.txt') # check if it's a file
True
>>> os.path.isfile('stuff')
False
>>> os.path.isdir('hello.txt') # check if it's a directory
False
>>> os.path.isdir('stuff')
True
>>> os.path.exists('hello.txt') # check if it's anything
True
>>> os.path.exists('stuff')
True
>>>
>>> # this joins with '\\' on windows and '/' on most other systems
>>> path = os.path.join('stuff', 'hello-world.txt')
>>> path
'stuff/hello-world.txt'
>>> with open(path, 'w') as f:
... # now this goes to the stuff folder we created
... print("Hello World!", file=f)
...
>>> os.listdir('stuff') # create a list of everything in stuff
['hello-world.txt']
>>>
Mix a list of things.
import random
print("Enter things to mix, and press Enter without typing",
"anything when you're done.")
things = []
while True:
thing = input("Next thing: ")
if thing == "":
break
things.append(thing)
random.shuffle(things)
print("After mixing:")
for thing in things:
print(thing)
Measure how long it takes for the user to answer a question.
The %.2f
rounds to 2 decimals, and you can find more formatting
tricks here.
import time
start = time.time()
answer = input("What is 1 + 2? ")
end = time.time()
difference = end - start
if answer == '3':
print("Correct! That took %.2f seconds." % difference)
else:
print("That's not correct...")
Wait a given number of seconds.
import sys
import time
answer = input("How long do you want to wait in seconds? ")
waitingtime = float(answer)
if waitingtime < 0:
print("Error: cannot wait a negative time.", file=sys.stderr)
sys.exit(1)
print("Waiting...")
time.sleep(waitingtime)
print("Done!")
Check what a path points to.
import os
import sys
print("You are currently in %s." % os.getcwd())
while True:
path = input("A path, or nothing at all to quit: ")
if path == '':
# We could just break out of the loop, but I'll show how
# this can be done with sys.exit. The difference is that
# break only breaks the innermost loop it is in, and
# sys.exit ends the whole program.
sys.exit()
if os.path.isfile(path):
print("It's a file!")
elif os.path.isdir(path):
print("It's a folder!")
elif os.path.exists(path):
# i have no idea when this code would actually run
print("Interesting, it exists but it's not a file or a folder.")
else:
print("I can't find it :(", file=sys.stderr)
Python's standard library has many awesome modules and I just can't tell about each and every module I use here. Here's some of my favorite modules from the standard library. Don't study them one by one, but look into them when you think you might need them. When reading the documentation it's usually easiest to find what you are looking for by pressing Ctrl+F in your web browser, and then typing in what you want to search for.
- argparse: a full-featured command-line argument parser
- collections, functools and itertools: handy utilities
- configparser: load and save setting files
- csv: store comma-separated lines in files
- json: yet another way to store data in files and strings
- textwrap: break long text into multiple lines
- warnings: like exceptions, but they don't interrupt the whole program
- webbrowser: open a web browser from Python
There are also lots of awesome modules that don't come with Python. You can search for those on the Python package index, or PyPI for short. It's often better to find a library that does something difficult than to spend a lot of time trying to do it yourself.
I recommend reading the official documentation about installing modules from PyPI. If you're using Linux, then also read the "Installing into the system Python on Linux" section at the bottom.
- Most modules are files on our computers, but some of them are built
in to Python. We can use modules in our projects by importing them,
and after that using
modulename.variable
to get a variable from the module. - Some of the most commonly used modules are random, sys, math, time and os.
- Avoid creating
.py
files that have the same name as a name of a module you want to use. - Python comes with many modules, and we can install even more modules if we want to.
TODO: exercises
So far we have made programs that ask the user to enter a string, and we also know how to convert that to an integer.
text = input("Enter something: ")
number = int(text)
print("Your number doubled:", number*2)
That works.
Enter a number: 3
Your number doubled: 6
But that doesn't work if the user does not enter a number.
Enter a number: lol
Traceback (most recent call last):
File "/some/place/file.py", line 2, in <module>
number = int(text)
ValueError: invalid literal for int() with base 10: 'lol'
So how can we fix that?
In the previous example we got a ValueError. ValueError is an exception. In other words, ValueError is an error that can occur in our program. If an exception occurs, the program will stop and we get an error message. The interactive prompt will display an error message and keep going.
>>> int('lol')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: invalid literal for int() with base 10: 'lol'
>>>
Exceptions are classes.
>>> ValueError
<class 'ValueError'>
>>>
We can also create exceptions. We won't get an error message by doing that, but we'll use this for displaying our own error messages later.
>>> the_problem = ValueError('oh no')
>>> the_problem
ValueError('oh no',)
>>>
If we need to try to do something and see if we get an exception, we
can use try
and except
. This is also known as catching the
exception.
>>> try:
... print(int('lol'))
... except ValueError:
... print("Oops!")
...
Oops!
>>>
The except part doesn't run if the try part succeeds.
>>> try:
... print("Hello World!")
... except ValueError:
... print("What the heck? Printing failed!")
...
Hello World!
>>>
ValueError is raised when something gets an invalid value, but the
value's type is correct. In this case, int
can take a string as an
argument, but the string needs to contain a number, not lol
. If
the type is wrong, we will get a TypeError instead.
>>> 123 + 'hello'
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: unsupported operand type(s) for +: 'int' and 'str'
>>>
Exceptions always interrupt the code even if we catch them. Here the
print never runs because it's after the error but inside the try
block. Everything after the try block runs normally.
>>> try:
... 123 + 'hello'
... print("This doesn't get printed.")
... except TypeError:
... print("Oops!")
...
Oops!
>>>
Does an except ValueError
also catch TypeErrors?
>>> try:
... print(123 + 'hello')
... except ValueError:
... print("Oops!")
...
Traceback (most recent call last):
File "<stdin>", line 2, in <module>
TypeError: unsupported operand type(s) for +: 'int' and 'str'
>>>
No, it doesn't. But maybe we could except for both ValueError and TypeError?
>>> try:
... int('lol')
... except ValueError:
... print('wrong value')
... except TypeError:
... print('wrong type')
...
wrong value
>>> try:
... 123 + 'hello'
... except ValueError:
... print('wrong value')
... except TypeError:
... print('wrong type')
...
wrong type
>>>
Seems to be working.
We can also also catch multiple exceptions by catching a tuple of exceptions:
>>> try:
... 123 + 'hello'
... except (ValueError, TypeError):
... print('wrong value or type')
...
wrong value or type
>>> try:
... int('lol')
... except (ValueError, TypeError):
... print('wrong value or type')
...
wrong value or type
>>>
Catching Exception
will catch all errors. We'll learn more about why
it does that in a moment.
>>> try:
... 123 + 'hello'
... except Exception:
... print("Oops!")
...
Oops!
>>> try:
... int('lol')
... except Exception:
... print("Oops!")
...
Oops!
>>>
It's also possible to catch an exception and store it in a variable.
Here we are catching an exception that Python created and storing it in
our_error
.
>>> try:
... 123 + 'hello'
... except TypeError as e:
... our_error = e
...
>>> our_error
TypeError("unsupported operand type(s) for +: 'int' and 'str'",)
>>> type(our_error)
<class 'TypeError'>
>>>
Do not do things like this:
try:
# many lines of code
except Exception:
print("Oops! Something went wrong.")
There's many things that can go wrong in the try
block. If something
goes wrong all we have is an oops message that doesn't tell us which
line caused the problem. This makes fixing the program really annoying.
If we want to catch exceptions we need to be specific about what exactly
we want to catch and where instead of catching everything we can in the
whole program.
There's nothing wrong with doing things like this:
try:
with open('some file', 'r') as f:
content = f.read()
except OSError: # we can't read the file but we can work without it
content = some_default_content
Usually catching errors that the user has caused is also a good idea:
import sys
text = input("Enter a number: ")
try:
number = int(text)
except ValueError:
print("'%s' is not a number." % text, file=sys.stderr)
sys.exit(1)
print("Your number doubled is %d." % (number * 2))
Now we know how to create exceptions and how to handle errors that Python creates. But we can also create error messages manually. This is known as raising an exception and throwing an exception.
Raising an exception is easy. All we need to do is to type raise
and then an exception we want to raise:
>>> raise ValueError("lol is not a number")
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: lol is not a number
>>>
Of course, we can also raise an exception from a variable.
>>> oops = ValueError("lol is not a number")
>>> raise oops
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: lol is not a number
>>>
If we define a function that raises an exception and call it we'll notice that the error message also says which functions we ran to get to that error.
>>> def oops():
... raise ValueError("oh no!")
...
>>> def do_the_oops():
... oops()
...
>>> do_the_oops()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 2, in do_the_oops
File "<stdin>", line 2, in oops
ValueError: oh no!
>>>
If our code was in a file we would also see the line of code that raised the error.
Back in the module chapter we learned to display error
messages by printing to sys.stderr
and then calling sys.exit(1)
, so
when should we use that and when should we raise an exception?
Exceptions are meant for programmers, so if we are writing something
that other people will import we should use exceptions. If our program
is working like it should be and the user has done something wrong,
it's usually better to use sys.stderr
and sys.exit
.
You may have more or less exceptions than I have if your Python is newer or older than mine, but they should be mostly similar.
Exception
βββ ArithmeticError
β βββ FloatingPointError
β βββ OverflowError
β βββ ZeroDivisionError
βββ AssertionError
βββ AttributeError
βββ BufferError
βββ EOFError
βββ ImportError
β βββ ModuleNotFoundError
βββ LookupError
β βββ IndexError
β βββ KeyError
βββ MemoryError
βββ NameError
β βββ UnboundLocalError
βββ OSError
β βββ BlockingIOError
β βββ ChildProcessError
β βββ ConnectionError
β β βββ BrokenPipeError
β β βββ ConnectionAbortedError
β β βββ ConnectionRefusedError
β β βββ ConnectionResetError
β βββ FileExistsError
β βββ FileNotFoundError
β βββ InterruptedError
β βββ IsADirectoryError
β βββ NotADirectoryError
β βββ PermissionError
β βββ ProcessLookupError
β βββ TimeoutError
βββ ReferenceError
βββ RuntimeError
β βββ NotImplementedError
β βββ RecursionError
βββ StopAsyncIteration
βββ StopIteration
βββ SyntaxError
β βββ IndentationError
β βββ TabError
βββ SystemError
βββ TypeError
βββ ValueError
β βββ UnicodeError
β βββ UnicodeDecodeError
β βββ UnicodeEncodeError
β βββ UnicodeTranslateError
βββ Warning
βββ BytesWarning
βββ DeprecationWarning
βββ FutureWarning
βββ ImportWarning
βββ PendingDeprecationWarning
βββ ResourceWarning
βββ RuntimeWarning
βββ SyntaxWarning
βββ UnicodeWarning
βββ UserWarning
Catching an exception also catches everything that's under it in this
tree. For example, catching OSError
catches errors that we typically
get when processing files, and catching Exception catches
all of these errors. You don't need to remember this tree, running
help('builtins')
should display a larger tree that this is a part of.
There are also a few exceptions that are not in this tree like SystemExit and KeyboardInterrupt, but most of the time we shouldn't catch them. Catching Exception doesn't catch them either.
- Exceptions are classes and they can be used just like all other classes.
- ValueError and TypeError are some of the most commonly used exceptions.
- The
try
andexcept
keywords can be used for attempting to do something and then doing something else if we get an error. This is known as catching exceptions. - It's possible to raise exceptions with the
raise
keyword. This is also known as throwing exceptions. - Raise exceptions if they are meant to be displayed for programmers and
use
sys.stderr
andsys.exit
otherwise.
Keep asking a number from the user until it's entered correctly.
while True:
try:
number = int(input("Enter a number: "))
break
except ValueError:
print("That's not a valid number! Try again.")
print("Your number doubled is:", number * 2)
This program allows the user to customize the message it prints by modifying a file the greeting is stored in, and it can create the file for the user if it doesn't exist already. This example also uses things from the file chapter, the function defining chapter and the module chapters.
# These are here so you can change them to customize the program
# easily.
default_greeting = "Hello World!"
filename = "greeting.txt"
import sys
def askyesno(question):
while True:
answer = input(question + ' (y or n) ')
if answer == 'Y' or answer == 'y':
return True
if answer == 'N' or answer == 'n':
return False
def greet():
with open(filename, 'r') as f:
for line in f:
print(line.rstrip('\n'))
try:
greet()
except OSError:
print("Cannot read '%s'!" % filename, file=sys.stderr)
if askyesno("Would you like to create a default greeting file?"):
with open(filename, 'w') as f:
print(default_greeting, file=f)
greet()
When I was getting started in Python I learned to make classes for tkinter GUI's before I understood how they work. Everything I did with classes worked, but I didn't understand how. Hopefully you'll first learn to understand classes, and then learn to use them.
Python comes with many classes that we know already.
>>> str
<class 'str'>
>>> int
<class 'int'>
>>> list
<class 'list'>
>>> dict
<class 'dict'>
>>>
Calling these classes as if they were functions makes a new instance
of them. For example, str()
makes a str
instance, also known as a
string.
>>> str()
''
>>> int()
0
>>> list()
[]
>>> dict()
{}
>>>
We can also get an instance's class with type()
:
>>> type('')
<class 'str'>
>>> type(0)
<class 'int'>
>>> type([])
<class 'list'>
>>> type({})
<class 'dict'>
>>>
Let's say that we make a program that processes data about websites.
With a custom class, we're not limited to str
, int
and other classes
Python comes with. Instead we can define a Website class, and make
Websites and process information about websites directly. Defining our
own types like this is called object-orientated programming.
In Python, pass
does nothing.
>>> pass
>>>
Let's use it to define an empty class.
>>> class Website:
... pass
...
>>> Website
<class '__main__.Website'>
>>>
The pass
is needed here, just like when defining functions that do
nothing.
Note that I named the class Website
, not website
. This way we know
that it's a class. Built-in classes use lowercase names (like str
instead of Str
) because they are faster to type, but use CapsWord
names for your classes.
Now we can make a Website instance by calling the class.
>>> github = Website()
>>> github
<__main__.Website object at 0x7f36e4c456d8>
>>> type(github)
<class '__main__.Website'>
>>>
We can say that github
is "a Website instance", "a Website
object" or "a Website". All of these mean the same thing.
Now we can attach more information about github to our Website.
>>> github.url = 'https://github.com/'
>>> github.founding_year = 2008
>>> github.free_to_use = True
>>>
We can also access the information easily.
>>> github.url
'https://github.com/'
>>> github.founding_year
2008
>>> github.free_to_use
True
>>>
As you can see, our Website is mutable, like lists are, not immutable like strings are. We can change the website in-place without creating a new Website.
url
, founding_year
and free_to_use
are not variables, they are
attributes. More specifically, they are instance attributes.
The biggest difference is that we need to use a dot for setting and
getting values of attributes, but we don't need that with variables.
Modules also use instance attributes for accessing their content. For
example, when we do random.randint
, random
is a module instance and
randint
is one of its attributes.
If we make another Website, does it have the same url
, founding_year
and free_to_use
?
>>> effbot = Website()
>>> effbot.url
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Website' object has no attribute 'url'
>>>
It doesn't. We'd need to define the attributes for effbot also.
The attributes are stored in a dictionary called __dict__
. It's not
recommended to use it for code that needs to be reliable, but it's a
handy way to see which attributes the instance contains.
>>> github.__dict__
{'free_to_use': True,
'founding_year': 2008,
'url': 'https://github.com/'}
>>> effbot.__dict__
{}
>>>
What happens if we set an attribute of the Website
class to some value
instead of doing that to an instance?
>>> Website.is_online = True
>>> Website.is_online
True
>>>
Seems to be working, but what happened to the instances?
>>> github.is_online
True
>>> effbot.is_online
True
>>>
What was that? Setting Website.is_online
to a value also set
github.is_online
and effbot.is_online
to that value!
Actually, is_online
is still not in github's or effbot's
__dict__
. github and effbot get that attribute directly from
the Website
class.
>>> github.__dict__
{'free_to_use': True,
'founding_year': 2008,
'url': 'https://github.com/'}
>>> effbot.__dict__
{}
>>>
Website.is_online
is Website
's class attribute, and in Python you can
access class attributes through instances also, so in this case
github.is_online
points to Website.is_online
. That can be
confusing, which is why it's not recommended to use class attributes like
this. Use instance attributes instead, e.g. github.is_online = True
.
Let's define a function that prints information about a website.
>>> def website_info(website):
... print("URL:", website.url)
... print("Founding year:", website.founding_year)
... print("Free to use:", website.free_to_use)
...
>>> website_info(github)
URL: https://github.com/
Founding year: 2008
Free to use: True
>>>
Seems to be working. We should be able to get information about all
websites, so maybe we should attach the website_info
function to the
Website class?
>>> Website.info = website_info
>>> Website.info(github)
URL: https://github.com/
Founding year: 2008
Free to use: True
>>>
It's working, but Website.info(github)
is a lot of typing, so
wouldn't github.info()
be much better?
>>> github.info()
URL: https://github.com/
Founding year: 2008
Free to use: True
>>>
What the heck happened? We didn't define a github.info
, it just
magically worked!
Website.info
is our website_info
function, so github.info
should also be the same function. But Website.info
takes a website
argument, which we didn't give it when we called github.info()
!
But is github.info
the same thing as Website.info
?
>>> Website.info
<function website_info at 0x7f36e4c39598>
>>> github.info
<bound method website_info of <__main__.Website object at 0x7f36e4c456d8>>
>>>
It's not.
Instead, github.info
is a method. If we set a function as a
class attribute, the instances will have a method with the same name.
Methods are "links" to the class attribute functions. So
Website.info(github)
does the same thing as github.info()
,
and when github.info()
is called it automatically gets
github
as an argument.
In other words, Class.method(instance)
does the same thing as
instance.method()
. This also works with built-in classes, for
example 'hello'.lower()
is same as str.lower('hello')
.
Maybe we could define a method when we make the class instead of adding it later?
>>> class Website:
... def info(self): # self will be github
... print("URL:", self.url)
... print("Founding year:", self.founding_year)
... print("Free to use:", self.free_to_use)
...
>>> github = Website()
>>> github.url = 'https://github.com/'
>>> github.founding_year = 2008
>>> github.free_to_use = True
>>> github.info()
URL: https://github.com/
Founding year: 2008
Free to use: True
>>>
It's working. The self
argument in Website.info
was github
.
You could call it something else too such as me
, this
or instance
,
but use self
instead. Other Python users have gotten used to it, and
the official style guide recommends it also.
We still need to set url
, founding_year
and free_to_use
manually.
Maybe we could add a method to do that?
>>> class Website:
... def initialize(self, url, founding_year, free_to_use):
... self.url = url
... self.founding_year = founding_year
... self.free_to_use = free_to_use
... def info(self):
... print("URL:", self.url)
... print("Founding year:", self.founding_year)
... print("Free to use:", self.free_to_use)
...
>>> github = Website()
>>> github.initialize('https://github.com/', 2008, True)
>>> github.info()
URL: https://github.com/
Founding year: 2008
Free to use: True
>>>
That works. The attributes we defined in the initialize method are also
available in the info method. We could also access them directly from
github
, for example with github.url
.
But we still need to call github.initialize
. In Python, there's
a "magic" method that runs when we create a new Website by calling the
Website class. It's called __init__
and it does nothing by default. If
our __init__
method takes other arguments than self we can call the
class with arguments and they will be given to __init__
. Like this:
>>> class Website:
... def __init__(self, url, founding_year, free_to_use):
... self.url = url
... self.founding_year = founding_year
... self.free_to_use = free_to_use
... def info(self):
... print("URL:", self.url)
... print("Founding year:", self.founding_year)
... print("Free to use:", self.free_to_use)
...
>>> github = Website('https://github.com/', 2008, True)
>>> github.info()
URL: https://github.com/
Founding year: 2008
Free to use: True
>>>
Classes have many other magic methods too, but I'm not going to cover them in this tutorial.
Don't do this:
class MyProgram:
def __init__(self):
print("Hello!")
word = input("Enter something: ")
print("You entered " + word + ".")
program = MyProgram()
You should avoid using things like print
and input
in the __init__
method. The __init__
method should be simple and it should just set
things up.
Usually you shouldn't use a class if you're only going to make one
instance of it, and you don't need a class either if you're only going
to have one method. In this example MyProgram
has only one method and
only one instance.
Make functions instead, or just write your code without any functions if it's short enough for that. This program does the same thing and it's much more readable:
print("Hello!")
word = input("Enter something: ")
print("You entered " + word + ".")
- Object-orientated programming is programming with custom data types. In Python that means using classes and instances.
- Use CapsWords for class names and lowercase_words_with_underscores for other names. This makes it easy to see which objects are classes and which objects are instances.
- Calling a class as if it was a function makes a new instance of it.
foo.bar = baz
setsfoo
's attributebar
tobaz
.- Use class attributes for functions and instance attributes for other things.
- Functions as class attributes can be accessed as instance methods.
They get their instance as the first argument. Call that
self
when you define the method. __init__
is a special method, and it's ran when a new instance of a class is created. It does nothing by default.- Don't use classes if your code is easier to read without them.
If you want to learn more advanced techniques, you can also read this section. Most of the techniques explained here are great when you're working on a large project, and your code would be really repetitive without these things.
You can experiment with these things freely, but please don't use these techniques just because you know how to use them. Prefer the simple techniques from the Basics part instead when possible. Simple is better than complex.
- Handy data types.
- Advanced stuff with functions.
- Magic methods.
- Iterables, iterators and generators.
You can star this tutorial. Starring is free for you, but to tells me and other people that you like this tutorial. Please give it a Star βοΈ