Lecture 10 - Lists, Mutability Flashcards

1
Q

Explain what is the difference of using .sort( ) and sorted( ) with a list object

A

.sort( ) is a method that mutates the original list object

[IN]: L1 = [2, 7, 3, 8, 9, 1, 4]
[IN]: L1.sort()
[IN]: print(L1)
[OUT]: [1, 2, 3, 4, 7, 8, 9]

sorted( ) is a function that does not mutate the original list object
~~~
[IN]: L2 = [2, 7, 3, 8, 9, 1, 4]
[IN]: L3 = sorted(L2)
[IN]: print(L3)
[OUT]: [1, 2, 3, 4, 7, 8, 9]
[IN]: print(L2)
[OUT]: [2, 7, 3, 8, 9, 1, 4]
~~~

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
2
Q

Explain the difference between mutable and immutable objects, and give an example for each one.

A

Mutable objects, like lists, can be altered after they have been created.
Alternatively, immutable objects, like tuples, str, int, cannot be altered once they have been created. If we want to change these, we need to create new objects are re-bind the variables (names) to to it.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
3
Q

Explain how to combine the elements from the list object new_ls into a single string, with each element being separated from the next one by a space.

[IN]: new_ls = ['A', 'stream', 'of',
'bright', 'points', 'flowing', 'from',
'left', 'to', 'right', 'and', 'combining',
'to', 'transform', 'into', 'a', 'solid',
'object']
A

Use the .join() method

[IN] phrase = " ".join(new_ls)
[IN] print(phrase)
[OUT]: 'A stream of bright points flowing 
from left to right and combining 
to transform into a solid object'
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
4
Q

Explain why we would not need a return statement for the function below that is intended to mutate a list by raising each element to the power of two.
~~~
def square_list(L):
for i in range(len(L)):
L[i] = L[i] ** 2
~~~

A

As the purpose of the function is to mutate the original list object, there is no need to bind the mutated list (function output) to a new object, hence we do not need the return.

We can just let the function mutate its elements, by means of a for loop and value assignment, and then print or call the list object to check its effect.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
5
Q

If we have data with a large number of composing elements, e.g. sales records from an online shop, then what would be the advantage of storing this in a mutable (e.g. list) vs immutable (e.g. tuple) object?

A

Suppose we want to change one element from this large dataset.
If we would use an immutable object, then we would need to copy the data to a new object since these kind of object cannot be altered after creation.
If we would use a mutable object, then we can simply change the records need by indexing/slicing, without having to create copies from the dataset.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
6
Q

Explain what is the output from code below.
~~~
L = [1,2,3,4]
i = 0
for e in L:
L.append(i)
i += 1
print(L)
~~~

A

It results in an infite loop. Each iteration e moves to the next element from L, but list object L also becomes one element larger. Effectively, e will never reach the end of L.

The key point here is that we are iterating over a mutating object and this needs attention as it may result in undesired behaviour.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
7
Q

Explain what will be the output from the last line of code. Will it be [1, 2, 3, 4, 5, 6] or [1, 2, 3, 7, 4, 5, 6] ?
~~~
[IN]: L1 = [1, 2, 3]
[IN]: L2 = [4, 5, 6]
[IN]: L3 = L1 + L2; print(L3)
[OUT]: [1, 2, 3, 4, 5, 6]
[IN]: L1.append(7);print(L1)
[OUT]: [1, 2, 3, 7]
[IN]: print(L3)
~~~

A

[OUT]: [1, 2, 3, 4, 5, 6]
Concatenating (+ operator) list objects uses copies from the lists being concatenated instead of references.

That is why after we mutate L1, L3 does not suffer any side effect.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
8
Q

Explain the output and execution flow of code below.
~~~
L = [1, 2, 3, 4]
for e in L:
L = L + L
print(L)
~~~

A

OUT: [1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4]

Think about the stack frame for variables e and L.

L mutates over the loop, but e keeps its binding to the original [1, 2, 3, 4] object hence it does not result in an infinite loop.

Concatante uses copies instead of references.

2nd iteration through the loop (see image below)

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
9
Q

Explain what will be the three outputs below. Think about object equality and mutability.
~~~
[IN]: L = [4,5,6];print(id(L))
[OUT]: ?
[IN]: L.append(8);print(id(L))
[OUT]: ?
[IN]: L.clear();print(id(L))
[OUT]: ?
~~~

A
[OUT]: 1888763712960
[OUT]: 1888763712960
[OUT]: 1888763712960

Outputs (ids) are the same since we are mutating over the same list object.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly
10
Q

Explain what will be the three outputs below. Think about object equality and mutability.
~~~
[IN]: L = [4,5,6];print(id(L))
[OUT]: ?
[IN]: L.append(8);print(id(L))
[OUT]: ?
[IN]: L = [];print(id(L))
[OUT]: ?
~~~

A
[OUT]: 1888763685312
[OUT]: 1888763685312
[OUT]: 1888763922624

The first two ids are the same since we are mutating the same list object.

The last id will be different since we created a new empy list object.

How well did you know this?
1
Not at all
2
3
4
5
Perfectly