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
If you have trouble with this tutorial please tell me about it and I'll make this tutorial better. If you like this tutorial, please give it a star.
You may use this tutorial freely at your own risk. See LICENSE.