2.4. Dictionaries in Python#

2.4.1. A dictionary is a Mapping#

In Python, a dictionary is a data structure that represents a collection of key-value pairs. It is also known as an associative array, hash map, or hash table in other programming languages. Dictionaries are mutable, which means you can modify their contents after they are created. They are particularly useful when you need to store and retrieve data based on a unique identifier (the key) [Downey, 2015, Python Software Foundation, 2024].

In a dictionary, each key is unique and maps to a specific value. The keys must be immutable data types (e.g., strings, numbers, tuples), while the values can be of any data type (e.g., strings, numbers, lists, other dictionaries).

Example: Creating a dictionary in Python:

# Method 1: Using curly braces and colons to define key-value pairs
person = {"Name": "John",
          "Age": 35,
          "City": "Calgary"
          }

# Method 2: Using the dict() constructor
person = dict(name="John", age=35, city=" Calgary ")

# Method 3: Using a list of tuples and the dict() constructor
person_info = [("Name", "John"), ("Age", 35), ("City", "Calgary")]
person = dict(person_info)
print(person)
{'Name': 'John', 'Age': 35, 'City': 'Calgary'}
../_images/key_value_pairs.png

Fig. 2.3 A visual representation of the defined dictionary .#

You can access values in the dictionary by using their corresponding keys:

# Create a dictionary representing a person's information
person = {"Name": "John",
          "Age": 35,
          "City": "Calgary"}

# Access and print specific values from the dictionary
print(f'Name = {person["Name"]}')
print(f'Age = {person["Age"]}')
print(f'City = {person["City"]}')
Name = John
Age = 35
City = Calgary

You can also modify, add, or remove key-value pairs in the dictionary:

# Modify the value associated with the "Age" key
person["Age"] = 31

# Add a new key-value pair to the dictionary
person["Occupation"] = "Engineer"

# Remove a key-value pair from the dictionary using the "City" key
del person["City"]

# Print the updated dictionary
print(person)
{'Name': 'John', 'Age': 31, 'Occupation': 'Engineer'}
../_images/key_value_pairs_updated.png

Fig. 2.4 A visual representation of the updated dictionary .#

Note - pprint in Python

The pprint (pretty-print) module in Python is used to provide a more readable and aesthetically pleasing representation of data structures, such as lists, dictionaries, and nested collections. It is particularly useful when working with complex or deeply nested data, as it formats the output in a way that is easier to understand at a glance.

Key Features:

  • Improved Readability: By organizing data into a structured and indented format, pprint makes it easier to spot relationships and hierarchies within the data.

  • Configurable Width: You can specify the maximum width of the output, which helps to prevent lines from becoming too long and wrapping awkwardly.

  • Handling of Recursive Data: pprint can handle recursive data structures, ensuring that references are printed in a way that avoids infinite loops or overly verbose outputs.

Useful Links:

  • pprint Documentation: The official documentation for the pprint module in Python.

  • pprint on PyPI: PyPI page for the pprint module, including installation instructions and additional information.

2.4.2. Iterating Over Dictionaries#

In Python, you can loop through dictionaries using various methods to access and work with their key-value pairs. Here are some common ways to loop through dictionaries [Downey, 2015, Python Software Foundation, 2024]:

2.4.2.1. Looping through keys using for loop#

Example:

# Create a dictionary with key-value pairs representing a person's information
my_dictionary = {"Name": "John", "Age": 35, "City": "Calgary", "Occupation": "Engineer"}

# Iterate through the keys in the dictionary and print each key-value pair
for key in my_dictionary:
    print(key, ':', my_dictionary[key])
Name : John
Age : 35
City : Calgary
Occupation : Engineer

Explanations:

  1. my_dictionary = {"Name": "John", "Age": 35, "City": "Calgary", "Occupation": "Engineer"}: This line initializes a Python dictionary named my_dictionary with key-value pairs. Each key represents a piece of information about a person, such as “Name,” “Age,” “City,” and “Occupation.” The associated values are specific to this individual.

  2. for key in my_dictionary:: This line initiates a for loop that iterates through the keys of the dictionary my_dictionary. During each iteration, it assigns the current key to the variable key.

    • key: This variable holds the current key from the dictionary during each iteration.

2.4.2.2. Looping through keys using keys() method#

Example:

# Iterate through the keys of the dictionary using the keys() method and print each key-value pair
for key in my_dictionary.keys():
    print(key, ':', my_dictionary[key])
Name : John
Age : 35
City : Calgary
Occupation : Engineer

Explanations:

for key in my_dictionary.keys():: This line initiates a for loop that iterates through the keys of the dictionary my_dictionary. In each iteration, it assigns the current key to the variable key.

  • key: This variable holds the current key from the dictionary during each iteration.

2.4.2.3. Looping through values using values() method#

Example:

# Iterate through the values of the dictionary using the values() method and print each value
for value in my_dictionary.values():
    print(value)
John
35
Calgary
Engineer

Explanations:

for value in my_dictionary.values():: This line initiates a for loop that iterates through the values of the dictionary my_dictionary. In each iteration, it assigns the current value to the variable value.

  • value: This variable holds the current value from the dictionary during each iteration.

2.4.2.4. Looping through key-value pairs using items() method#

Example:

# Iterate through key-value pairs in the dictionary using the items() method and print each key and value
for key, value in my_dictionary.items():
    print(key, ':', value)
Name : John
Age : 35
City : Calgary
Occupation : Engineer

Explanations:

for key, value in my_dictionary.items():: This line initiates a for loop that iterates through the items of the dictionary my_dictionary. In each iteration, it unpacks the key-value pair into two variables, key and value.

  • key: This variable holds the current key in the dictionary during each iteration.

  • value: This variable holds the corresponding value associated with the current key.

2.4.2.5. Using a list comprehension to manipulate the dictionary#

# Create a dictionary with key-value pairs representing a person's information
my_dictionary = {"Name": "John", "Age": 35, "City": "Calgary", "Occupation": "Engineer"}

# For example, let's create a new dictionary where age is increased by 1 for each person
# Iterate through key-value pairs in the original dictionary, updating the "Age" key if found
new_dictionary = {key: value + 1 if key == 'Age' else value for key, value in my_dictionary.items()}

# Print the updated dictionary
print(new_dictionary)
{'Name': 'John', 'Age': 36, 'City': 'Calgary', 'Occupation': 'Engineer'}

Explanations:

new_dictionary = {key: value + 1 if key == 'Age' else value for key, value in my_dictionary.items()}: In this line, a new dictionary named new_dictionary is created using a dictionary comprehension. This comprehension iterates through the key-value pairs of the my_dictionary dictionary using the items() method. For each key-value pair, it checks if the key is “Age.” If the key is “Age,” it increments the value by 1; otherwise, it retains the original value. This results in a new dictionary with the same key-value pairs as my_dictionary, except that the “Age” value has been increased by 1.

  • key: value + 1 if key == 'Age' else value: This expression updates the value based on the condition. If the key is “Age,” it adds 1 to the value; otherwise, it keeps the original value.

2.4.2.6. Combining loops with conditionals#

# Iterate through the dictionary and check each key-value pair
# For example, let's print only the key-value pairs where the value is a string
for key, value in my_dictionary.items():
    # Check if the value is a string using isinstance
    if isinstance(value, str):
        # If it's a string, print the key and value
        print(key, ':', value)
Name : John
City : Calgary
Occupation : Engineer

These are some common techniques to loop through dictionaries in Python. Depending on your specific use case, you can choose the most appropriate method for your needs. Remember that dictionaries are unordered, so the order in which the elements are iterated may not match the order in which they were inserted.

2.4.3. Reverse lookup#

In Python, dictionaries support forward lookup (accessing the value using the key), but by design, they do not directly support reverse lookup (accessing the key using the value). However, you can achieve reverse lookup functionality using various approaches. Here are a couple of common methods to perform reverse lookup on a dictionary [Downey, 2015, Python Software Foundation, 2024]:

Example: Consider the following dictionary: my_dictionary = {'a': 1, 'b': 2, 'c': 3, 'd': 2}

../_images/Reverse_lookup.png

Fig. 2.5 A visual representation of the defined dictionary .#

  • Method 1: Using a loop to find keys for a given value

# Define a function to perform reverse lookup in a dictionary
def reverse_lookup(d, value):
    keys = []  # Initialize an empty list to store keys
    for key, val in d.items():  # Iterate through key-value pairs in the dictionary
        if val == value:  # Check if the current value matches the target value
            keys.append(key)  # Append the current key to the list if it's a match
    return keys  # Return the list of keys

# Create a dictionary with key-value pairs
my_dictionary = {'a': 1, 'b': 2, 'c': 3, 'd': 2}

# Call the reverse_lookup function to find keys with a specific value
result = reverse_lookup(my_dictionary, 2)

# Print the result, which is a list of keys with the specified value
print(result)  # Output: ['b', 'd']
['b', 'd']

Explanations:

  1. Function Definition (reverse_lookup): Defines a function that searches a dictionary (d) for keys associated with a specific value (value). It returns a list of keys that match the value.

  2. Example Usage: Creates a dictionary (my_dictionary) with key-value pairs. Calls reverse_lookup with my_dictionary and the value 2 to find keys with that value. Prints the resulting list (['b', 'd']).

  • Method 2: Using a dictionary comprehension (Python 3.8+)

# Define a function to perform reverse lookup in a dictionary
def reverse_lookup(d, value):
    keys = []  # Initialize an empty list to store keys
    for key, val in d.items():  # Iterate through key-value pairs in the dictionary
        if val == value:  # Check if the current value matches the target value
            keys.append(key)  # Append the current key to the list if it's a match
    return keys  # Return the list of keys

# Create a dictionary with key-value pairs
my_dictionary = {'a': 1, 'b': 2, 'c': 3, 'd': 2}

# Call the reverse_lookup function to find keys with a specific value
result = reverse_lookup(my_dictionary, 2)

# Print the result, which is a list of keys with the specified value
print(result)  # Output: ['b', 'd']
['b', 'd']

Explanations:

  1. Function (reverse_lookup): Defines a function that searches a dictionary (d) for keys associated with a specific value. It returns a list of keys that match the value.

  2. Example Usage: Creates a dictionary my_dictionary with keys 'a', 'b', 'c', 'd' and values 1, 2, 3, 2 respectively. Calls reverse_lookup with my_dictionary and 2 to find keys with that value. Prints the resulting list ['b', 'd'].

Note

In both methods, the function reverse_lookup takes a dictionary d and a value as input and returns a list of keys that have the specified value. Note that these methods will return all the keys corresponding to the given value. If you expect only one key-value pair with the given value, you can modify the function to return just the first key found or handle multiple keys as needed.

Keep in mind that if you anticipate the need for reverse lookups frequently, you may want to consider maintaining a separate reverse dictionary that maps values to keys, especially if the dictionary is large, to optimize lookup performance.

2.4.4. Dictionaries and lists#

In Python, dictionaries and lists are two different built-in data structures that serve different purposes and have distinct characteristics. Let’s explore their differences and how they can be used [Downey, 2015, Python Software Foundation, 2024]:

2.4.4.1. Dictionaries:#

  • Dictionaries are unordered collections of key-value pairs.

  • They use a hash table to provide efficient lookup and insertion.

  • Keys must be unique and immutable (e.g., strings, numbers, or tuples), while values can be of any data type.

  • You can access, modify, add, and delete elements using their keys.

  • Dictionaries are typically used when you need to map keys to corresponding values and perform fast lookups based on those keys.

2.4.4.2. Lists:#

  • Lists are ordered collections of elements.

  • Elements in a list can be of any data type and can be duplicated.

  • You can access, modify, add, and delete elements in a list using their positions (indices).

  • Lists are useful when you need to store multiple items in a specific order, and you may perform operations like appending, slicing, or sorting.

2.4.4.3. Combining dictionaries and lists#

  • You can use dictionaries and lists together to represent more complex data structures.

  • For example, you can have lists of dictionaries or dictionaries containing lists.

Example: Example of a list of dictionaries:

people = [{'Name': 'John', 'Age': 30, 'Occupation': 'Engineer'},
          {'Name': 'Jane', 'Age': 25, 'Occupation': 'Doctor'},
          {'Name': 'Mike', 'Age': 35, 'Occupation': 'Teacher'}
          ]

Explanations:

  • people: This is a list that contains three dictionaries. Each dictionary represents a person’s information, and these dictionaries are enclosed in square brackets, indicating that they are elements of a list.

  • Each dictionary within the list represents a person with three key-value pairs:

    1. 'Name': This key stores the person’s name as a string. For example, the first dictionary has 'Name': 'John', indicating that the person’s name is John.

    2. 'Age': This key stores the person’s age as an integer. For instance, the first dictionary has 'Age': 30, which means the person’s age is 30 years old.

    3. 'Occupation': This key stores the person’s occupation as a string. For example, the first dictionary has 'Occupation': 'Engineer', indicating that the person’s occupation is an engineer.

So, the people list contains information about three individuals: John, Jane, and Mike, with their respective names, ages, and occupations stored in dictionaries within the list. This data structure is useful for organizing and storing information about multiple individuals or entities in a structured format.

Example: Example of a dictionary containing lists:

my_data = {'Numbers': [1, 2, 3, 4, 5],
           'Fruits': ['Apple', 'Banana', 'Orange'],
           'Colors': ['Red', 'Blue', 'Green']
           }

Explanations:

  • my_data: This is a dictionary that contains three key-value pairs, enclosed within curly braces {}.

  • Each key in the dictionary represents a category, and each value is a list of items associated with that category. Here are the details of each key-value pair:

    1. 'Numbers': This key represents a category related to numbers. The associated value is a list [1, 2, 3, 4, 5], which contains five integers.

    2. 'Fruits': This key represents a category related to fruits. The associated value is a list ['Apple', 'Banana', 'Orange'], which contains three strings, each representing a different fruit.

    3. 'Colors': This key represents a category related to colors. The associated value is a list ['Red', 'Blue', 'Green'], which contains three strings, each representing a different color.

This data structure allows you to organize and store data in a hierarchical manner. It’s a convenient way to group related information, and you can access items within each category by using the corresponding key. For example, you can access the list of numbers as my_data['Numbers'], which would return [1, 2, 3, 4, 5]. Similarly, you can access the list of fruits as my_data['Fruits'], and the list of colors as my_data['Colors'].

2.4.4.4. Similarities and Differences Between Dictionaries and Lists#

Dictionaries and lists are two common data structures in programming, often used to store and manage collections of items. They have distinct characteristics that make them suitable for different purposes. Here are the similarities and differences between dictionaries and lists [Downey, 2015, Python Software Foundation, 2024]:

Similarities:

  1. Collection of Items: Both dictionaries and lists are used to store multiple items or elements in a single data structure.

  2. Mutable: Both dictionaries and lists are mutable, meaning you can modify their contents after creation. You can add, update, or remove items from them.

  3. Iteration: You can iterate over the items in both dictionaries and lists using loops.

  4. Mixed Data Types: Both data structures can store elements of various data types, such as strings, numbers, booleans, and even other data structures.

Differences:

  1. Access by Key vs. Index: The primary difference between dictionaries and lists is how you access their elements. In a dictionary, elements are stored as key-value pairs, and you access an element using its associated key. In a list, elements are stored in a linear fashion, and you access an element using its index (position).

  2. Ordering: Dictionaries in Python (prior to Python 3.7) were unordered collections, meaning the order of elements was not guaranteed. However, starting from Python 3.7, dictionaries maintain insertion order. Lists, on the other hand, maintain their order by default.

  3. Duplication: In a dictionary, keys must be unique, while values can be duplicated. In a list, elements can be duplicated.

  4. Usage: Dictionaries are often used when you have a set of unique keys that map to corresponding values, allowing for efficient lookups and retrieval of values based on keys. Lists are used when you need an ordered collection of items, and you primarily access them based on their position.

  5. Size: Dictionaries are typically more memory-intensive than lists because they store both keys and values. Lists only store the values themselves.

  6. Complexity: The time complexity for accessing an element in a dictionary is typically O(1) on average due to the hash-based indexing of keys. In a list, accessing an element by index has a time complexity of O(1), but searching for an element by value takes O(n) time on average, where n is the number of elements.

Example:

Dictionary example:

student_info = {'Name': 'John',
                'Age': 25,
                'Grade': 'A'
                }
  • Structure: This is a dictionary where each element is a key-value pair. The keys are 'Name', 'Age', and 'Grade', and their corresponding values are 'John', 25, and 'A', respectively.

  • Accessing Elements: You access elements by their keys. For example, student_info['Name'] retrieves the value 'John'.

  • Adding Elements: You can add a new key-value pair by assigning a value to a new key, like student_info['City'] = 'Calgary'.

List example:

fruits = ['Apple', 'Banana', 'Orange']
  • Structure: This is a list, which is an ordered collection of elements. Each element is accessed by its index, starting from 0.

  • Accessing Elements: You access elements by their index. For example, fruits retrieves the value 'Banana'.

  • Adding Elements: You can add an element to the end of the list using append(), like fruits.append('Grape').

Iterating Through a Dictionary

for key, value in student_info.items():
    print(key, '=', value)
Name = John
Age = 25
Grade = A
  • Explanation: The items() method returns a view object that displays a list of a dictionary’s key-value tuple pairs. This allows you to iterate over both keys and values simultaneously.

Iterating Through a List

for fruit in fruits:
    print(fruit)
Apple
Banana
Orange
  • Explanation: This loop iterates over each element in the list fruits. Each element is accessed directly, and the loop variable fruit takes on the value of each element in turn.

Additional Notes

  • Mutability: Both dictionaries and lists are mutable, meaning their contents can be changed after creation. This is demonstrated in the example by adding elements to both data structures.

  • Efficiency: The efficiency of operations can vary between dictionaries and lists. For instance, looking up a value by key in a dictionary is generally faster than searching for an element in a list, especially as the size of the data structure grows.