This lesson is inspired by the Programming in Python lessons from the Software Carpentry organization. The main sections of the lesson are below.
We saw a bit about variables and their values in first lesson, and we continue here with some variables related to rock samples collected on a recent field excursion. For a given rock sample, the standard for assigning a sample identification value is to use the format
PE-LO-NU-YR
, wherePE
is for the first and last initials of the person that collected the sample,LO
is a two-letter abbreviation for the sample location,NU
is the sample number on that excursion, andYR
is the last two digits of the year in which the sample was collected. We can store this information and some additional information about the samples in Python as follows:SampleID = 'DW-NP-48-16' SampleNumber = 48 SampleWeightLbs = 6.89 SampleRockType = 'Mica schist'
Here we have 4 values assigned to variables related to a single rock sample. Each variable has a unique name and they can store different types of data.n
We can explore the different types of data stored in variables using the
type()
function. There are 4 basic data types in Python as shown in the table below.
Data type name | Data type | Example |
---|---|---|
int |
Whole integer values | 4 |
float |
Decimal values | 3.1415 |
str |
Character strings | 'Hei' |
bool |
True/false values | True |
As you will see, the data types are important because some are not compatible with one another.
>>> type(SampleNumber) int >>> type(SampleWeightLbs) float >>> type(SampleRockType) str
Ok. So we can get the data types for variables in Python using the
type()
function, but why does this matter? The main reason is that some types of data are compatible with one another, while others are not.>>> SampleWeightLbs * 25 # Estimate weight of 25 samples 172.25 >>> SampleWeightLbs / 2.2 # Calculate sample weight in pounds 3.1318181818181814
In Python 3 is it generally not a problem to multiply or divide
int
andfloat
variables. As you might expect, you can also multiply them by themselves.>>> SampleWeightLbs * SampleWeightLbs 47.4721
The trouble can arise when trying to use mathematical operations with incompatible data types.
>>> SampleWeightLbs + SampleID --------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-20-75e50db6ddf4> in <module>() ----> 1 SampleWeightLbs + SampleID TypeError: unsupported operand type(s) for +: 'float' and 'str'
Here we get a
TypeError
because Python does not know how we are expected to combine a string of characters (SampleID
) with a decimal value number (SampleWeightLbs
).Interestingly, some operations combining numbers and
str
values will work.>>> SampleID * 3 'DW-NP-48-16DW-NP-48-16DW-NP-48-16'
Here the values in the
SampleID
variable are simply repeated 3 times.One of the nice options in IPython is that you can see which variables are in memory and their values by typing
%whos
.>>> %whos Variable Type Data/Info ----------------------------------- SampleID str DW-NP-48-16 SampleWeightLbs float 6.89 SampleNumber int 48 SampleRockType str Mica schist
%whos
is an IPython magic command that will not work in a standard Python interpreter window. We will see other magic commands as we learn more Python. They're useful!
As we've seen above, my recent field excursion involved collecting (at least) 48 rock samples. Rather than having individual variables for each of those samples, we can store many related values in a collection. The simplest type of collection in Python is a list.
Let's first create a list of selected
SampleID
values.>>> SampleIDs = ['DW-NP-03-16', 'DW-NP-12-16', 'DW-NP-33-16', 'DW-NP-48-16'] >>> print(SampleIDs) ['DW-NP-03-16', 'DW-NP-12-16', 'DW-NP-33-16', 'DW-NP-48-16'] >>> type(SampleIDs) list
Here we have a list of 4
SampleID
values in a list calledSampleIDs
. As you can see, thetype()
function recognizes this as a list. Lists can be created using the square brackets ([
and]
), with commas separating the values in the list. nTo access an individual value in the list we need to use an index value. An index value is a number that refers to a given position in the list. Let's check out the first value in our list as an example:
>>> print(SampleIDs[1]) 'DW-NP-12-16'
Wait, what? This is the second value in the list we've created, what is wrong? As it turns out, Python (and many other programming languages) start values stored in collections with the index value 0. Thus, to get the value for the first item in the list, we must use index 0.
>>> print(SampleIDs[0]) 'DW-NP-03-16'
OK, that makes sense now, but it may take some getting used to...
We can find the length of a list using the
len()
function.>>> len(SampleIDs) 4
Just as expected, there are 4 values in our list and
len(SampleIDs)
returns a value of 4.If we know the length of the list, we can now use it to find the value of the last item in the list, right?
>>> print(SampleIDs[4]) --------------------------------------------------------------------------- IndexError Traceback (most recent call last) <ipython-input-34-946b174fe444> in <module>() ----> 1 print(SampleIDs[4]) IndexError: list index out of range
What, an
IndexError
? That's right, since our list starts with index 0 and has 4 values, the index of the last item in the list islen(SampleIDs) - 1
. That isn't ideal, but fortunately there's a nice trick in Python to find the last item in a list.>>> print(SampleIDs) ['DW-NP-03-16', 'DW-NP-12-16', 'DW-NP-33-16', 'DW-NP-48-16'] >>> print(SampleIDs[-1]) 'DW-NP-48-16' >>> print(SampleIDs[-4]) 'DW-NP-03-16'
Yes, in Python you can go backwards through lists by using negative index values. Index -1 gives the last value in the list and index
-len(SampleIDs)
would give the first. Of course, you still need to keep the index values within their ranges.>>> print(SampleIDs[-5]) --------------------------------------------------------------------------- IndexError Traceback (most recent call last) <ipython-input-38-ac2327014588> in <module>() ----> 1 print(SampleIDs[-5]) IndexError: list index out of range
Another nice feature of lists is that they are mutable, meaning that the values in a list that has been defined can be modified. Consider a list of the rock types corresponding to the sample IDs in the
SampleIDs
list.>>> SampleRockTypes = ['Augen gneiss', 'Leucogranite', 'Quartzite', 'Mica schst'] >>> print(SampleRockTypes) ['Augen gneiss', 'Leucogranite', 'Quartzite', 'Mica schst']
Now as we saw before, the rock types for sample DW-NP-48-16 should be 'Mica schist', not 'Mica schst'. Fortunately, this is an easy fix. We simply replace the value at the corresponding location in the list with the correct definition.
>>> SampleRockTypes[3] = 'Mica schist' >>> print(SampleRockTypes) ['Augen gneiss', 'Leucogranite', 'Quartzite', 'Mica schist']
Lists also do not need to have only one type of data. Let's consider that in addition to having a list of each sample ID, sample number, rock type, etc. we would like to have a list of all of the values for sample 'DW-NP-48-16'.
>>> Sample48 = [SampleID, SampleNumber, SampleWeightLbs, SampleRockType] >>> print(Sample48) ['DW-NP-48-16', 48, 6.89, 'Mica schist']
Here we have one list with 3 different type of data in it. We can confirm this using the
type()
function.>>> type(Sample48) list >>> type(Sample48[0]) # The sample ID str >>> type(Sample48[1] # The sample number int >>> type(Sample48[2]) # The sample weight float
Finally, we can add and remove values from lists to change their lengths. Let's consider that we no longer want to include the first value in the
SampleIDs
list.>>> print(SampleIDs) ['DW-NP-03-16', 'DW-NP-12-16', 'DW-NP-33-16', 'DW-NP-48-16'] >>> del SampleIDs[0] >>> print(SampleIDs) ['DW-NP-12-16', 'DW-NP-33-16', 'DW-NP-48-16']
del
allows values in lists to be removed. It can also be used to delete values from memory in Python. If we would instead like to add a few samples to theSampleIDs
list, we can do so as follows.>>> SampleIDs.append('DW-NP-27-16') >>> SampleIDs.append('DW-NP-51-16') >>> print(SampleIDs) ['DW-NP-12-16', 'DW-NP-33-16', 'DW-NP-48-16', 'DW-NP-27-16', 'DW-NP-51-16']
As you can see, we add values one at a time using
SampleIDs.append()
.list.append()
is called a method in Python, which is a function that works for a given data type (a list in this case). We'll see a bit more about these below.
Python is one of a number of computer programming languages that are called "object-oriented languages". The simple explanation is that we can consider the variables that we define to be "objects" that can contain both data known as attributes and a specific set of functions (methods). The previous sentence could take quite some time to understand by itself, but using an example the concept of "objects" is much easier to understand.
Let's consider our list
SampleIDs
. As we know, we already have data in the listSampleIDs
, and we can modify that data using built-in methods such asSampleIDs.append()
. We can also do other things such as count the number of times a value occurs in a list, or where it occurs.>>> SampleIDs.count('DW-NP-27-16') # The count method counts the number of occurences of a value 1 >>> SampleIDs.index('DW-NP-27-16') # The index method gives the index value of an item in a list 3
The good news here is that our selected sample ID is only in the list once. Should we need to modify it for some reason, we also now know where it is in the list (index
3
).There are two other common methods for lists that we need to see. First, there is the
.sort()
method, used to sort values in a list. As you can see from when we appended the additional two sample IDs earlier, our list no longer has sample IDs in increasing order. We can fix that.>>> SampleIDs.sort() >>> print(SampleIDs) ['DW-NP-12-16', 'DW-NP-27-16', 'DW-NP-33-16', 'DW-NP-48-16', 'DW-NP-51-16']
Yay, it works! A common mistake when sorting lists is to do something like
SampleIDs = SampleIDs.sort()
. Do not do this! When sorting with.sort()
theNone
value is returned (this is why there is no screen ouput when runningSampleIDs.sort()
). If you then assign the output ofSampleIDs.sort()
toSampleIDs
you will sort the list, but then overwrite its contents with the returned valueNone
. This means you've deleted the list contents (!).The
.reverse()
method works the same way.>>> SampleIDs.reverse() # Notice no output here... >>> print(SampleIDs) ['DW-NP-51-16', 'DW-NP-48-16', 'DW-NP-33-16', 'DW-NP-27-16', 'DW-NP-12-16']
As you can see, the list has been reversed using the
.reverse()
method, but there is no screen output when this occurs. Again, if you were to assign that output toSampleIDs
the list would get reversed, but the contents would then be assignedNone
.