So here's a simple question. What's the complexity of adding an element at the end of a linked list? The naive answer usually is 'constant time'. But let's check the usual implementations.
Let's assume we have the structure:
typedef struct _node {
int value;
struct _node *next;
} node;
Implementation 1:
void add (node **head, int value)
{
node *new_node, *ptr;
if (!head) return;
new_node = (node *)malloc (sizeof (node));
new_node->next = NULL;
new_node->value = value;
ptr = *head;
while (ptr && ptr->next) ptr = ptr->next;
if (ptr)
ptr->next = new_node;
else
*head = new_node;
}
This implementation is clearly NOT in constant time because, well, we have to start from the head, get all the way to the end of the list and add the element there. Time: O(n). At least. More later.
The obvious fix would be to also have the tail of the list. To make the tail available as well. So let's expand our data structures with a 'list' structure that will contain the head and the tail.
typedef struct {
node *head;
node *tail;
} list;
Implementation 2:
void add (list *l, int value)
{
node *new_node, *ptr;
if (!l) return;
new_node = (node *)malloc (sizeof (node));
new_node->next = NULL;
new_node->value = value;
if (l->tail) {
l->tail->next = new_node;
l->tail = new_node;
} else {
l->head = l->tail = new_node;
}
}
Great! Problem solved! We have constant time for adding. Don't we? Begginer's luck, we're safe at bay!
...
...
Oh wait...
...
...
What is the execution time of malloc?
Oh sh**.
Good, now that we found out that it's not so easy to do that, we can say goodbye to the idea that a dynamically allocated linked list does not have the possibility to add an element in O(1) simply because we have to allocate the element. In academia it's a good and nice O(1) solution, in RealLife™ it's not. So what do we do?
The solutions I found so far are not perfect by far. First solution would be to have something like a smart pool that will give us the elements in a more or less constant time (one idea would be to have a buffer of more node structures, and then one could perform a circular search from the last 'allocated' node). Another solution would be to not do dynamic allocation at all, but that defeats the purpose of having a linked list in the first place.
One reminder. Never forget that just because you have a one liner doesn't mean that the execution time for that line is constant. And memory allocation can be one of the most critical operation for the performance of your program
No comments:
Post a Comment