Lecture 13 - Linked Lists Flashcards

1
Q

linked list

A

struct node {
int value;
struct node *next_ptr;
};
- create three and put them together in memory
- let each one point to the next, and the last have a NULL pointer
- this creates a linked list

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

code for basic linked list

A

struct node *one_ptr = NULL;
struct node *two_ptr = NULL;
struct node *three_ptr = NULL;

one_ptr = malloc(sizeof(struct node))’
one_ptr->value = 12;
// same for two/three
one_prt->next_ptr = two_ptr;
two_ptr->next_ptr = three_ptr;
three_ptr->next_ptr = NULL;

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

why that basic code has too many pointers

A
  • in practice, we would do the previous example without using so many pointers
  • for instance, we can refer to any element within any of the structures via the first structure
  • ex. one_ptr->next_ptr->value is the value of the second element in the list
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
4
Q

forming a linked list: growing “forward”

A
  • adding to end of list
  • use one pointer to refer to the “head” of the list
  • use second pointer to refer to the “tail” of the list
  • add every new structure to tail->next_ptr
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
5
Q

forming a linked list: growing “backward” (adding to the beginning)

A
  • adding to beginning of list
  • use one pointer to refer to the “head” of the list
  • use a temporary pointer to refer to a new structure
  • set temp->next_ptr = head
  • set head = temp
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
6
Q

special case for first node

A
  • the first thing we must to for LLs is allocate the head
  • if (head == NULL) {
    head = malloc(sizeof(struct node));
    assert(head != NULL);
    head->next_ptr = NULL;
    }
  • in the forward growing case, we then set the tail to the head
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
7
Q

forward growing initial setup

A
  • declare head and tail variables
  • set tail equal to head
  • they will both be pointing to the same next value
  • the next value’s next_ptr is NULL
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
8
Q

forward growing (step two)

A
  • make the next_ptr of head/tail point to the value you want to add
  • added value’s next-ptr is NULL
  • allocate size for added value
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
9
Q

forward growing (step three)

A
  • change it so the tail is equal to the final added value
  • tail = tail->next_ptr;
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
10
Q

reverse growing (initial setup)

A
  • create head and temp variables
  • head’s next_ptr points to a value, temp is NULL
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
11
Q

reverse growing (step two)

A
  • temp points to a value now
  • head points to the same thing
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
12
Q

reverse growing (step three)

A
  • make the head point to the same next_ptr as the temp
  • temp->next_ptr = head;
  • head = temp;
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
13
Q

traversing a linked list

A
  • usually, you do not know how many structures are in a linked list
  • have to “traverse” it to find an item or do work on the structures
  • you can traverse a linked list with one extra pointer
    p = head;
    while (p != NULL) {
    p->value++;
    p = p->next_ptr;
    }
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
14
Q

deleting a linked list

A
  • deletion of a linked list is a special case of the traversal process
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
15
Q

common mistakes with deleting a linked list

A
  • freeing the memory of the pointer before saving it to another variable
How well did you know this?
1
Not at all
2
3
4
5
Perfectly
16
Q

functions to simplify list management

A
  • writing code to do operations on list can be repetitive, tedious, error prone
  • usually a good idea to encapsulate the functionality into functions to create, delete, insert, and spend new structures
17
Q

create_node() example code

A
  • allocate a new node, check the malloc() return value and set the fields
    struct node *create_node(int new_value) {
    struct node *temp = NULL;
    temp = malloc(sizeof(struct node));
    assert(temp != NULL);
    temp->value = new_value;
    temp->next_ptr = NULL;
    return temp;
    }
18
Q

linked list structs w/more variables

A
  • normally a structure in a linked list contains many more elements than just a single value and a list pointer
19
Q

doubly linked list

A
  • without the head can’t traverse a linked list/move nodes
  • a doubly linked list contains two pointers
  • a “next” pointer and “previous” pointer
  • goes forward and backward with pointers
20
Q

example of doubly linked list declaration

A

include <stdio.h></stdio.h>

#include <malloc.h>
#include <assert.h></assert.h></malloc.h>

struct double_l {
int value;
struct double_l *next_ptr;
struct double_l *prev_ptr;
};

21
Q

doubly linked list use example

A

int main() {
struct double_l *ptr = NULL;
ptr = malloc(sizeof(struct double_l));
assert(ptr != NULL);
ptr->next_ptr = NULL;
ptr->prev_ptr = NULL;
ptr->value = 15;
return ptr->value;
}

22
Q

what’s wrong with this?
p = head
while (p != NULL) {
free(p);
p = p->next_ptr;
}

A
  • the node is being freed before moving to the next node
  • the pointer b becomes invalid after being freed, so this results in undefined behavior
  • must save next pointer before freeing the current node
23
Q

what’s wrong with this?
p = head;
while (p != NULL) {
struct node *tmp = p->next;
free(p);
p = tmp;
}

A
  • correct way to delete a linked list
  • stores pointer in tmp before freeing the current node
  • free(p) deallocate s the memory but tmp still points to the next node in the list
  • p = tmp updates p to point to he next node, letting the loop continue