Welcome to Locomotive Engineer on Exercism's Python Track.
If you need help running the tests or submitting your code, check out HELP.md
.
If you get stuck on the exercise, check out HINTS.md
, but try and solve it without using those first :)
Unpacking refers to the act of extracting the elements of a collection, such as a list
, tuple
, or dict
, using iteration.
Unpacked values can then be assigned to variables within the same statement, which is commonly referred to as Multiple assignment.
The special operators *
and **
are often used in unpacking contexts and with multiple assignment.
`*<variable_name>` and `**<variable_name>` should not be confused with `*` and `**`. While `*` and `**` are used for multiplication and exponentiation respectively, `*<variable_name>` and `**<variable_name>` are used as packing and unpacking operators.
In multiple assignment, the number of variables on the left side of the assignment operator (=
) must match the number of values on the right side.
To separate the values, use a comma ,
:
>>> a, b = 1, 2
>>> a
1
If the multiple assignment gets an incorrect number of variables for the values given, a ValueError
will be thrown:
>>> x, y, z = 1, 2
ValueError: too many values to unpack (expected 3, got 2)
Multiple assignment is not limited to one data type:
>>> x, y, z = 1, "Hello", True
>>> x
1
>>> y
'Hello'
>>> z
True
Multiple assignment can be used to swap elements in lists
.
This practice is pretty common in sorting algorithms.
For example:
>>> numbers = [1, 2]
>>> numbers[0], numbers[1] = numbers[1], numbers[0]
>>> numbers
[2, 1]
Since tuples
are immutable, you can't swap elements in a tuple
.
The examples below use `lists` but the same concepts apply to `tuples`.
In Python, it is possible to unpack the elements of list
/tuple
/dictionary
into distinct variables.
Since values appear within lists
/tuples
in a specific order, they are unpacked into variables in the same order:
>>> fruits = ["apple", "banana", "cherry"]
>>> x, y, z = fruits
>>> x
"apple"
If there are values that are not needed then you can use _
to flag them:
>>> fruits = ["apple", "banana", "cherry"]
>>> _, _, z = fruits
>>> z
"cherry"
Unpacking and assigning values from a list
/tuple
inside of a list
or tuple
(also known as nested lists/tuples), works in the same way a shallow unpacking does, but often needs qualifiers to clarify the values context or position:
>>> fruits_vegetables = [["apple", "banana"], ["carrot", "potato"]]
>>> [[a, b], [c, d]] = fruits_vegetables
>>> a
"apple"
>>> d
"potato"
You can also deeply unpack just a portion of a nested list
/tuple
:
>>> fruits_vegetables = [["apple", "banana"], ["carrot", "potato"]]
>>> [a, [c, d]] = fruits_vegetables
>>> a
["apple", "banana"]
>>> c
"carrot"
If the unpacking has variables with incorrect placement and/or an incorrect number of values, you will get a ValueError
:
>>> fruits_vegetables = [["apple", "banana"], ["carrot", "potato"]]
>>> [[a, b], [d]] = fruits_vegetables
ValueError: too many values to unpack (expected 1)
When unpacking a list
/tuple
you can use the *
operator to capture the "leftover" values.
This is clearer than slicing the list
/tuple
(which in some situations is less readable).
For example, we can extract the first element and then assign the remaining values into a new list
without the first element:
>>> fruits = ["apple", "banana", "cherry", "orange", "kiwi", "melon", "mango"]
>>> x, *last = fruits
>>> x
"apple"
>>> last
["banana", "cherry", "orange", "kiwi", "melon", "mango"]
We can also extract the values at the beginning and end of the list
while grouping all the values in the middle:
>>> fruits = ["apple", "banana", "cherry", "orange", "kiwi", "melon", "mango"]
>>> x, *middle, y, z = fruits
>>> y
"melon"
>>> middle
["banana", "cherry", "orange", "kiwi"]
We can also use *
in deep unpacking:
>>> fruits_vegetables = [["apple", "banana", "melon"], ["carrot", "potato", "tomato"]]
>>> [[a, *rest], b] = fruits_vegetables
>>> a
"apple"
>>> rest
["banana", "melon"]
Unpacking a dictionary is a bit different than unpacking a list
/tuple
.
Iteration over dictionaries defaults to the keys.
So when unpacking a dict
, you can only unpack the keys and not the values:
>>> fruits_inventory = {"apple": 6, "banana": 2, "cherry": 3}
>>> x, y, z = fruits_inventory
>>> x
"apple"
If you want to unpack the values then you can use the values()
method:
>>> fruits_inventory = {"apple": 6, "banana": 2, "cherry": 3}
>>> x, y, z = fruits_inventory.values()
>>> x
6
If both keys and values are needed, use the items()
method.
Using items()
will generate tuples with key-value pairs.
This is because of dict.items()
generates an iterable with key-value tuples
.
>>> fruits_inventory = {"apple": 6, "banana": 2, "cherry": 3}
>>> x, y, z = fruits_inventory.items()
>>> x
("apple", 6)
Packing is the ability to group multiple values into one list
that is assigned to a variable.
This is useful when you want to unpack values, make changes, and then pack the results back into a variable.
It also makes it possible to perform merges on 2 or more lists
/tuples
/dicts
.
Packing a list
/tuple
can be done using the *
operator.
This will pack all the values into a list
/tuple
.
>>> fruits = ("apple", "banana", "cherry")
>>> more_fruits = ["orange", "kiwi", "melon", "mango"]
# fruits and more_fruits are unpacked and then their elements are packed into combined_fruits
>>> combined_fruits = *fruits, *more_fruits
# If there is no * on to the left of the "=" the result is a tuple
>>> combined_fruits
("apple", "banana", "cherry", "orange", "kiwi", "melon", "mango")
# If the * operator is used on the left side of "=" the result is a list
>>> *combined_fruits_too, = *fruits, *more_fruits
>>> combined_fruits_too
['apple', 'banana', 'cherry', 'orange', 'kiwi', 'melon', 'mango']
Packing a dictionary is done by using the **
operator.
This will pack all key-value pairs from one dictionary into another dictionary, or combine two dictionaries together.
>>> fruits_inventory = {"apple": 6, "banana": 2, "cherry": 3}
>>> more_fruits_inventory = {"orange": 4, "kiwi": 1, "melon": 2, "mango": 3}
# fruits_inventory and more_fruits_inventory are unpacked into key-values pairs and combined.
>>> combined_fruits_inventory = {**fruits_inventory, **more_fruits_inventory}
# then the pairs are packed into combined_fruits_inventory
>>> combined_fruits_inventory
{"apple": 6, "banana": 2, "cherry": 3, "orange": 4, "kiwi": 1, "melon": 2, "mango": 3}
When you create a function that accepts an arbitrary number of arguments, you can use *args
or **kwargs
in the function definition.
*args
is used to pack an arbitrary number of positional (non-keyworded) arguments and
**kwargs
is used to pack an arbitrary number of keyword arguments.
Usage of *args
:
# This function is defined to take any number of positional arguments
>>> def my_function(*args):
... print(args)
# Arguments given to the function are packed into a tuple
>>> my_function(1, 2, 3)
(1, 2, 3)
>>> my_function("Hello")
("Hello")
>>> my_function(1, 2, 3, "Hello", "Mars")
(1, 2, 3, "Hello", "Mars")
Usage of **kwargs
:
# This function is defined to take any number of keyword arguments
>>> def my_function(**kwargs):
... print(kwargs)
# Arguments given to the function are packed into a dictionary
>>> my_function(a=1, b=2, c=3)
{"a": 1, "b": 2, "c": 3}
*args
and **kwargs
can also be used in combination with one another:
>>> def my_function(*args, **kwargs):
... print(sum(args))
... for key, value in kwargs.items():
... print(str(key) + " = " + str(value))
>>> my_function(1, 2, 3, a=1, b=2, c=3)
6
a = 1
b = 2
c = 3
You can also write parameters before *args
to allow for specific positional arguments.
Individual keyword arguments then have to appear before **kwargs
.
[Arguments have to be structured](https://www.python-engineer.com/courses/advancedpython/18-function-arguments/) like this:
`def my_function(<positional_args>, *args, <key-word_args>, **kwargs)`
If you don't follow this order then you will get an error.
>>> def my_function(a, b, *args):
... print(a)
... print(b)
... print(args)
>>> my_function(1, 2, 3, 4, 5)
1
2
(3, 4, 5)
Writing arguments in an incorrect order will result in an error:
>>>def my_function(*args, a, b):
... print(args)
>>>my_function(1, 2, 3, 4, 5)
Traceback (most recent call last):
File "c:\something.py", line 3, in <module>
my_function(1, 2, 3, 4, 5)
TypeError: my_function() missing 2 required keyword-only arguments: 'a' and 'b'
You can use *
to unpack a list
/tuple
of arguments into a function call.
This is very useful for functions that don't accept an iterable
:
>>> def my_function(a, b, c):
... print(c)
... print(b)
... print(a)
numbers = [1, 2, 3]
>>> my_function(*numbers)
3
2
1
Using *
unpacking with the zip()
function is another common use case.
Since zip()
takes multiple iterables and returns a list
of tuples
with the values from each iterable
grouped:
>>> values = (['x', 'y', 'z'], [1, 2, 3], [True, False, True])
>>> a, *rest = zip(*values)
>>> rest
[('y', 2, False), ('z', 3, True)]
Your friend Linus is a Locomotive Engineer who drives cargo trains between cities. Although they are amazing at handling trains, they are not amazing at handling logistics or computers. They would like to enlist your programming help organizing train details and correcting mistakes in route data.
This exercise could easily be solved using slicing, indexing, and various `dict` methods.
However, we would like you to practice packing, unpacking, and multiple assignment in solving each of the tasks below.
Your friend has been keeping track of each wagon identifier (ID), but they are never sure how many wagons the system is going to have to process at any given time. It would be much easier for the rest of the logistics program to have this data packaged into a unified list
.
Implement a function get_list_of_wagons()
that accepts an arbitrary number of wagon IDs.
Each ID will be a positive integer.
The function should then return
the given IDs as a single list
.
>>> get_list_of_wagons(1, 7, 12, 3, 14, 8, 5)
[1, 7, 12, 3, 14, 8, 5]
At this point, you are starting to get a feel for the data and how it's used in the logistics program. The ID system always assigns the locomotive an ID of 1, with the remainder of the wagons in the train assigned a randomly chosen ID greater than 1.
Your friend had to connect two new wagons to the train and forgot to update the system!
Now, the first two wagons in the train list
have to be moved to the end, or everything will be out of order.
To make matters more complicated, your friend just uncovered a second list
that appears to contain missing wagon IDs.
All they can remember is that once the new wagons are moved, the IDs from this second list
should be placed directly after the designated locomotive.
Linus would be really grateful to you for fixing their mistakes and consolidating the data.
Implement a function fix_list_of_wagons()
that takes two lists
containing wagon IDs.
It should reposition the first two items of the first list
to the end, and insert the values from the second list
behind (on the right hand side of) the locomotive ID (1).
The function should then return
a list
with the modifications.
>>> fix_list_of_wagons([2, 5, 1, 7, 4, 12, 6, 3, 13], [3, 17, 6, 15])
[1, 3, 17, 6, 15, 7, 4, 12, 6, 3, 13, 2, 5]
Now that all the wagon data is correct, Linus would like you to update the system's routing information.
Along a transport route, a train might make stops at a few different stations to pick up and/or drop off cargo.
Each journey could have a different amount of these intermediary delivery points.
Your friend would like you to update the systems routing dict
with any missing/additional delivery information.
Implement a function add_missing_stops()
that accepts a routing dict
followed by a variable number of keyword arguments.
These arguments could be in the form of a dict
holding one or more stops, or any number of stop_number=city
keyword pairs.
Your function should then return the routing dict
updated with an additional key
that holds a list
of all the added stops in order.
>>> add_missing_stops({"from": "New York", "to": "Miami"},
stop_1="Washington, DC", stop_2="Charlotte", stop_3="Atlanta",
stop_4="Jacksonville", stop_5="Orlando")
{"from": "New York", "to": "Miami", "stops": ["Washington, DC", "Charlotte", "Atlanta", "Jacksonville", "Orlando"]}
Linus has been working on the routing program and has noticed that certain routes are missing some important details.
Initial route information has been constructed as a dict
and your friend would like you to update that dict
with whatever might be missing.
Every route in the system requires slightly different details, so Linus would really prefer a generic solution.
Implement a function called extend_route_information()
that accepts two dicts
.
The first dict
contains the origin and destination cities the train route runs between.
The second dict
contains other routing details such as train speed, length, or temperature.
The function should return a consolidated dict
with all routing information.
The second `dict` can contain different/more properties than the ones shown in the example.
>>> extend_route_information({"from": "Berlin", "to": "Hamburg"}, {"length": "100", "speed": "50"})
{"from": "Berlin", "to": "Hamburg", "length": "100", "speed": "50"}
When Linus was surveying the wagon depot they noticed that the wagons were not getting stored in the correct order. In addition to an ID, each wagon has a color that corresponds to the type of cargo it carries. Wagons are stored in the depot in grids, where each column in the grid has wagons of the same color.
However, the logistics system shows lists
of wagons to be stored in the depot have their rows grouped by color.
But for the storage grid to work correctly, each row should have three different colors so that the columns align by color.
Your friend would like you to sort out the wagon depot lists
, so that the wagons get stored correctly.
Implement a function called fix_wagon_depot()
that accepts a list
of three items.
Each list
item is a sublist (or "row") that contains three tuples
.
Each tuple
is a (<wagon ID>, <wagon color>)
pair.
Your function should return a list
with the three "row" lists
reordered to have the wagons swapped into their correct positions.
>>> fix_wagon_depot([
[(2, "red"), (4, "red"), (8, "red")],
[(5, "blue"), (9, "blue"), (13,"blue")],
[(3, "orange"), (7, "orange"), (11, "orange")],
])
[
[(2, "red"), (5, "blue"), (3, "orange")],
[(4, "red"), (9, "blue"), (7, "orange")],
[(8, "red"), (13,"blue"), (11, "orange")]
]
- @meatball133
- @BethanyG
- @IsaacG