Engineering Computing

Lists

The list class defines an ordered set of elements. These elements can be of any class, and do not need to match within a list. Lists can be nested to create a list of lists. The basic syntax for creating a list of elements e\(x\) is [e1, e2, ..., en]. Consider the following list assignments:

int_list = [3, 9, 3, -4, 0]         # Duplication allowed
str_list = ["foo", "bar", "baz"]
com_list = [int_list, str_list]     # List of lists
mix_list = [8.41, "foo", [7]]       # Mixing element types

Accessing List Elements

Because the elements of a list have an order, they can be referred to via an index, a mapping of integers to elements. In Python, the first element in the list has index 0 and subsequent elements have indices of increasing values, 1, 2, 3, and so on. The syntax for accessing the element with index i of a list l is l[i]. For instance, elements from the previously defined lists can be accessed as follows:

int_list[0]                         # => 3
int_list[3]                         # => -4
str_list[2]                         # => "baz"
mix_list[2]                         # => [7]

Negative indices are used to access elements from the end of a list. For instance, for int_list above,

int_list[-1]                        # => 0
int_list[-2]                        # => -4

This is particularly useful when we want to access the last element of a list, which we see has index -1.

A selection of elements from a list can be accessed via slicing, which has the syntax l[start:stop] or l[start:stop:step]. For instance,

l = [0, 1, 2, 3, 4]
l[0:3]                              # => [0, 1, 2]
l[2:4]                              # => [2, 3]
l[0:-1]                             # => [0, 1, 2, 3] (no last item!)
l[0:]                               # => [0, 1, 2, 3, 4]
l[0::2]                             # => [0, 2, 4] (every two elements)

It is important to note that the slice does not include the stop index; rather, the slice’s last value is from index stop-1. As we see in the third slice example, this means the normal syntax for slicing through the final element (i.e., the element with index -1) does not include that element. To include the final element, leave off an index for stop, as shown in the fourth and fifth examples.

Mutability

Lists are mutable; that is, they can be mutated (changed). This is unlike most built-in types, which are immutable and cannot be changed. The mutability for frequently used built-in types is shown in tbl. 1.

Table 1: Mutability of commonly used built-in types.
Data Type Built-in Class Mutability
Numbers int, float, complex Immutable
Strings str Immutable
Tuples tuple Immutable
Booleans bool Immutable
Lists list Mutable
Dictionaries dict Mutable
Sets set Mutable

The mutability of lists allows us to change their elements. The syntax for assigning a new value v to an element with index i of a list l is l[i] = v. For instance,

l = ["Hello", "World", "!"]
l[1] = "Stranger"
print(l)

returns

['Hello', 'Stranger', '!']

Note that although strings are immutable, a list of strings is mutable. This means "Stranger" is not at the same location in memory as was "World".

Methods

Lists have several methods for mutating themselves, which are given in tbl. 2.

Table 2: Commonly used list methods for a list l.
Method Description
l.append(item) Append item to the end of l
l.clear() Remove all items from l
l.extend(iterable) Concatenate l with the contents of iterable
l.index(x[, start[, end]]) Return the index of the first instance of x in l[start:end]
l.insert(index, item) Insert item into l at index
l.pop(index) Return and remove the item at index
l.pop() Return and remove the last item
l.remove(item) Remove item’s first occurrence
l.reverse() Reverse the items of l
l.sort(key=None, reverse=False) Sort the items of l

For example, an element can be inserted into a list as follows:

l = ["zero", "one", "three"]
l.insert(2, "two")
print(l)

which returns

['zero', 'one', 'two', 'three']

When using most list methods, we often do not assign the returned value from the expression. This is because most of these expressions return a value of None. For instance, from the previous example,

print(l.insert(2, "two"))

returns

None

Such methods are simply operating on the original list object and do not return that object. This is a common idiom in Python programming, and many mutable classes behave similarly.

Example 1.3

Write a program that removes the second occurrence of the element 3 from the following list:

l = [1, 2, 3, 0, 3, 4, 3]

The remove() method might seem promising, but it only removes the first occurrence of the element. Instead, let’s identify the index of the second occurrence. The index(x[, start[, end]]) method allows us to identify the index of the first occurrence or the first occurrence between start and end. So our strategy is to find the index i_first of the first occurrence with index(), then narrow our search to the rest of the list after i_first to the end of the list, identifying the second index i_second. Finally, we can remove the element at i_second with the pop method.

The following program implements this strategy.

l = [1, 2, 3, 0, 3, 4, 3]
x = 3                               # element we are removing
i_first = l.index(x)                # first occurrence index
i_second = l.index(x, i_first+1)    # second occurrence index
l.pop(i_second)                     # removes second occurrence
print(f"l without second {x}: {l}")

This prints

l without second 3: [1, 2, 3, 0, 4, 3]

Online Resources for Section 1.6

No online resources.