7.6 Revisiting The Bag ADTTo illustrate the use of the linked list structure, we implement a new version of the Bag ADT, which we originally defined in Section 2.1. As a review, the definition for the Bag ADT is provided below: The Bag ADT
A Linked List ImplementationUp to this point, we have mainly used the Python list to implement a variety of abstract data types. The singly linked list can also be used to implement a wide range of abstract data types. The choice of the underlying data structure depends heavily on the efficiency of the various operations of the abstract data type being implemented. The ultimate goal is to provide the most efficient implementation possible. Thus, the Python list or vector is not always the best choice for implementing some abstract data types. The Bag ADT provides a simple example of how to implement an abstract data type using a singly linked list. The items in the bag will be stored in the individual nodes of the list, one item per node. To implement a new version of the Bag ADT, we must define a new class with all of the same methods as the original version. This version will be stored in a module named As a reminder, a linked list is not itself an abstract data type. Instead, system components (references and objects) are used as building blocks to construct the dynamic linear structure. Each time you implement an abstract data type using a linked list, you must include some of the same code each time. Storage ClassWe begin our discussion of the linked list implementation of the Bag ADT with the definition of a storage class used to represent the individual nodes. The node storage class will have the same structure as that used in the previous sections, but we use the name class BagListNode : def __init__(self, data) : self.data = data self.next = None ConstructorWe begin the implementation with the constructor. Each instance of the class will need two attributes: the head reference and the number of elements in the bag. class Bag : def __init__(self) : self._head = None self._size = 0 The
Bag SizeThe length operation returns the number of items contained in the bag. This method simply returns the value of the def __len__(self) : return self._size Question 7.6.1
What is the time-complexity for the length operation?
Select the correct answer below by clicking on the appropriate box.
Returning the value of an instance variable is a constant time operation.
Question 7.6.2
Had we not included the
Select the correct answer below by clicking on the appropriate box.
The number of items in the bag would have to be determined by counting the number of nodes in the linked list. This could only be done by traversing the list and counting the number of iterations required for the complete traversal.
Contains MethodThe def __contains__(self, target) : curNode = self._head while curNode and curNode.item != target : curNode = curNode.next return curNode is not None Question 7.6.3
Is the linked list search more efficient than the linear search performed on a Python list?
Select the correct answer below by clicking on the appropriate box.
Both require linear time.
Adding an ItemTo add an item to the bag, we will need to store the item in a node and link the node into the linked list. Since a bag has no specific ordering, we can prepend the new node onto the linked list using the same code from Section 7.4. The only difference is that we must also increment the item counter as new items are added. Here is the code for the def add(self, item) : newNode = BagListNode(item) newNode.next = self._head self._head = newNode self._size += 1 Question 7.6.4
What is the time-complexity for the add operation?
Select the correct answer below by clicking on the appropriate box.
We saw earlier that prepending a node onto a linked list can be done in constant time. The implementation of this method is basically the same other than incrementing the counter, which is itself a constant time operation.
Removing an ItemThe def remove(self, item): predNode = None curNode = self._head while curNode and curNode.item != item : predNode = curNode curNode = curNode.next # The item has to be in the bag to remove it. assert curNode is not None, "The item must be in the bag." # Unlink the node and return the item. self._size -= 1 if curNode is self._head : self._head = curNode.next else : predNode.next = curNode.next return curNode.item The Question 7.6.5
What would happen if we omit the statement <code> self._size = self._size - 1 </code> which decrements the bag size counter? The length operation would not provide the correct value when used after items have been removed from the bag. The value returned would not reflect the correct number of items in the bag. The New ImplementationThe new implementation of the Bag ADT using a singly linked list is shown in the source listing below: Program Listing
Comparing ImplementationsThe Python list and the linked list can both be used to manage the elements stored in a bag. Both implementations provide the same time-complexities for the various operations with the exception of the
In general, the choice between a linked list or a Python list depends on the application as both have advantages and disadvantages. The linked list is typically a better choice for those applications involving large amounts of dynamic data, data that changes quite often. If there will be a large number of insertions and/or deletions, the linked structure provides a fast implementation since large amounts of data do not have to be shifted as is required by the Python list. This is especially true when prepending items. On the other hand, the Python list is a better choice in those applications where individual elements must be accessed by position. This can be simulated with a linked list, but it requires a traversal of the list, resulting in a linear operation whereas the Python list only requires constant time. Question 7.6.6
Would the linked list be an ideal structure for implementing the Polygon ADT that was presented earlier this term? Since the Polygon is an indexed sequence in which the elements need to be accessed by position, the linked list would not be a good structure for implementing the Polygon ADT. The Python list was the best choice for the implementation.
|