Lecture 9 - Lambda Functions, Tuples, and Lists Flashcards

1
Q

Explain what is the output from the code below:
~~~
[IN]: L = [1,2,3]
[IN]: L.append(L)
[IN]: print(L is L[-1])
~~~

A

[OUT]: True

  1. L = [1,2,3] - First, a list L is created containing the elements 1, 2, and 3.
  2. L.append(L) - The list L is appended to itself. This creates a recursive data structure where the list contains a reference to itself as its last element. After this line, L becomes [1, 2, 3, [...]] where [...] is a reference to the list itself.
  3. print(L is L[-1]) - This line checks if L and L[-1] (the last element of L) are the same object using the is operator. Since L[-1] is a reference to L itself (due to the previous append operation), this comparison returns True.
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
2
Q

Explain what is the output from the code below:
~~~
[IN]: L1 = [[]]*2
[IN]: L2 = [[], []]
[IN]: for i in range(len(L1)):
[IN]: L1[i].append(i)
[IN]: L2[i].append(i)
[IN]: print(‘L1 =’, L1, ‘but’, ‘L2 =’, L2)
~~~

A

[OUT]: L1 = [[0, 1], [0, 1]] but L2 = [[0], [1]]
Because the first assignment statement creates a list with two elements, each of which is the same object, whereas the second assignment statement creates a list with two different objects, each of which is initially equal to an empty list.

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

Explain what is aliasing in Python

A

Aliasing is when two or more variables (names) refer to the same object.

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

Explain how to create a lamba function that tells if a number is even or not. Moreover, explain how to use it in the REPL.

A

lambda x: x%2 == 0

This will return True if x is even, or False otherwise. To use it, we can use the code below:

[IN]: (lambda x: x%2 == 0)(8)
[OUT]: True

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

Explain how a lamba function works

A

lamba parameters : expression
completar

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

How can a tuple be used to swap variables with less lines of code?
~~~
[IN]: x = 1
[IN]: y = 2
[IN]: temp = x
[IN]: x = y
[IN]: y = temp
~~~

A
[IN]: x = 1
[IN]: y = 2
[IN]: (x, y) = (y, x)

Right-hand side of line 3 is evaluated first and then its values are bound/assigned to x, y

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

Explain how can we return multiple values from a function? Give an example.

A

Use a tuple
A function can only return one output so we can define it to return a single tuple that contains multiple values within it.

def quotient_and_reminder(x, y):
    q = x // y
    r = x % y
    return (q, r)
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
8
Q

How to create a function that takes an undefined number of formal arguments?

A

Use *args when defining the function

def mean(*args):
    """
    Assumes at least one argument and all arguments are numbers. 
    Returns the mean of the arguments.
    """
    tot = 0
    for a in args:
        tot += a
    return tot/len(args)
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
9
Q

Explain how can we test for object equality

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

But objects of
immutable types cannot be modified after they are created. On the
other hand, objects of mutable types can be modified after they are
created

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

Explain what would be the output from running the script below.
~~~
def append_val(val, list_1 = []):
list_1.append(val)
print(list_1)

append_val(3)
append_val(4)
~~~

A
[OUT]: 3
[OUT]: = [3, 4]

This happens because, at function definition
time
, a new object of type list is created, with an initial value of the empty list. Each time append_val is invoked without supplying a value for the formal parameter list_1, the object created at function definition is bound to list_1, mutated, and then printed. So, the second call to append_val mutates and then prints a list that was already mutated by the first call to that function.

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

Explain what would be the output from running the script below.

def remove_dups(L1, L2):
    """Assumes that L1 and L2 are lists.
    Removes any element from L1 that also occurs in L2
    """
    for e1 in L1:
        if e1 in L2:
            L1.remove(e1)

L1 = [1,2,3,4]
L2 = [1,2,5,6]
remove_dups(L1, L2)
print('L1 =', L1)
A

[OUT]: L1 = [2, 3, 4]

During a for loop, Python keeps track of where it is in the list using an internal counter that is incremented at the end of each iteration. When the value of the counter reaches the current length of the list, the loop terminates.

In this case, the hidden counter starts out at 0, discovers that L1[0] is in L2, and removes it—reducing the length of L1 to 3. The counter is then incremented to 1, and the code proceeds to check if the value of L1[1] is in L2. Notice that this is not the original value of L1[1] (i.e., 2), but rather the current value of L1[1] (i.e., 3). Effectively, this means 2 in L1 was skipped and will not be removed.

Since 3 and 4 are not in L2, no more elements are removed from L1.

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

Does the second line from script below creates a copy of L1?
~~~
[IN]: L1 = [1, 2, 3, 4]
[IN]: L1_1 = L1
~~~

A

No, it only binds a new name L1_1 to the same list object [1, 2, 3, 4] to which L1 is already binded.

Note that this is aliasing.

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

Explain what is the output from running the script below. What kind of copy is created by line 3?
~~~
[IN]: L = [2]
[IN]: L1 = [L]
[IN]: L2 = L1.copy()
[IN]: L.append(3)
[IN]: print(f’L1 = {L1}, L2 = {L2}’)
~~~

A

[OUT]: L1 = [[2, 3]], L2 = [[2, 3]]

It produces a shallow copy from L1, meaning it creates a list and puts the same objects from L1 in this copy.
Effectively, if I mutate L1, then this will be reflected in L2.

L2 = L1.copy() or L2 = L1[:] produce the same effect in terms of copying L1.

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

Explain how can I produce a deep copy of L1 in line 3 and bind it to L2. Also, explain how this kind of copy is different from a shallow copy.
~~~
[IN]: L = [2]
[IN]: L1 = [L]
[IN]: L2 =_________________
[IN]: L.append(3)
[IN]: print(f’L1 = {L1}, L2 = {L2}’)
~~~

A
[IN] import copy
[IN]: L2 = copy.deepcopy(L1)

A deep copy creates a new list L2 and put copies of objects from L1 into it. This meas that if I mutate L1, then change will not be reflected into L2.

[OUT]: L1 = [[2, 3]], L2 = [[2]]

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

Explain the output from the script below
~~~
[IN]: L1 = [2]
[IN]: L2 = [L1, L1]
[IN]: L3 = copy.deepcopy(L2)
[IN]: L3[0].append(3)
[IN]: print(L3)
~~~

A

[OUT]: [[2, 3], [2, 3]]

copy.deepcopy makes one copy of L1
and uses it both times L1 occurs in L2

17
Q

Explain the output from the script below
~~~
[IN]: a = [[] for _ in range(10)]
[IN]: a
~~~

A

[OUT]: [[], [], [], [], [], [], [], [], [], []]
It use a list comprehension to generate a list containing 10 distinct (i.e., non-aliased) empty lists

18
Q

Explain what is a list comprehension

A

[ <expression> for item in iterable <if optional_condition> ]

continuar…