Refactoring head_list and tail_list

Simplifying head and tail list. Adding delete_node() to list API.
This commit is contained in:
Michael Tryby
2019-04-11 10:45:22 -04:00
parent af87edbafe
commit 3f37fdbff4
3 changed files with 34 additions and 46 deletions

View File

@@ -62,12 +62,7 @@ void delete_list(list_t *list)
while(list->head != NULL) { while(list->head != NULL) {
current = list->head; current = list->head;
list->head = current->next; list->head = current->next;
delete_node(list, current);
if (list->freeFn)
list->freeFn(current->data);
free(current->data);
free(current);
} }
free(list); free(list);
} }
@@ -119,47 +114,26 @@ void for_each_list(list_t *list, listIterator iterator)
} }
} }
void *head_list(list_t *list, bool removeFromList) list_node_t *head_list(list_t *list, bool removeFromList)
// //
// Warning: Caller is responsible for freeing the node->data returned. // Warning: When node is removed caller is responsible for freeing it.
// //
{ {
assert(list->head != NULL); assert(list->head != NULL);
list_node_t *node = list->head; list_node_t *node = list->head;
// Allocating and copying pointer to node data
void *element = (void *)malloc(list->elementSize);
memcpy(element, node->data, list->elementSize);
if(removeFromList) { if(removeFromList) {
// Disconnecting head node // Disconnecting head node
list->head = node->next; list->head = node->next;
list->logicalLength--; list->logicalLength--;
// Freeing pointer to node->data and node
free(node->data);
free(node);
} }
// Now element points to data formerly pointed to by node. Caller return node;
// is responsible for freeing both pointer to data and data.
return element;
} }
void *tail_list(list_t *list) list_node_t *tail_list(list_t *list)
//
// Warning: Caller is responsible for freeing the node->data returned.
//
{ {
assert(list->tail != NULL); assert(list->tail != NULL);
return list->tail;
list_node_t *node = list->tail;
// Allocating and copying pointer to node data
void *element = (void *)malloc(list->elementSize);
memcpy(element, node->data, list->elementSize);
// Pointer to element data gets returned. Caller is responsible
// for freeing pointer to data.
return element;
} }
int size_list(list_t *list) int size_list(list_t *list)
@@ -172,9 +146,17 @@ void *get_data(list_node_t *lnode)
return lnode->data; return lnode->data;
} }
void delete_node(list_t *list, list_node_t *lnode)
{
if (list->freeFn)
list->freeFn(lnode->data);
free(lnode->data);
free(lnode);
}
// //
// Iterator first/done/next operations // Iterator first/done/next operations provide containment for list abstraction
// http://www.cs.yale.edu/homes/aspnes/pinewiki/C(2f)Iterators.html // http://www.cs.yale.edu/homes/aspnes/pinewiki/C(2f)Iterators.html
// Accessed on April 11, 2019 // Accessed on April 11, 2019
// //

View File

@@ -59,11 +59,17 @@ void append_list(list_t *list, void *element);
*/ */
int size_list(list_t *list); int size_list(list_t *list);
/** /**
@brief Returns pointer to list node's data. @brief Returns pointer to list node's data.
*/ */
void *get_data(list_node_t *lnode); void *get_data(list_node_t *lnode);
/**
@brief Frees memory associated with a list node.
*/
void delete_node(list_t *list, list_node_t *lnode);
/** /**
@brief Calls the supplied iterator function with the data element of each @brief Calls the supplied iterator function with the data element of each
@@ -74,12 +80,12 @@ void for_each_list(list_t *list, listIterator iterator);
/** /**
@brief Returns the head of the list (optionally removing it at the same time). @brief Returns the head of the list (optionally removing it at the same time).
*/ */
void *head_list(list_t *list, bool removeFromList); list_node_t *head_list(list_t *list, bool removeFromList);
/** /**
@brief Returns the tail of the list. @brief Returns the tail of the list.
*/ */
void *tail_list(list_t *list); list_node_t *tail_list(list_t *list);
/** /**

View File

@@ -116,25 +116,21 @@ BOOST_FIXTURE_TEST_CASE(test_string_list, FixtureStrings) {
BOOST_FIXTURE_TEST_CASE(test_head_list, FixtureStrings) { BOOST_FIXTURE_TEST_CASE(test_head_list, FixtureStrings) {
void *temp = head_list(list, true); BOOST_CHECK(check_string(get_string_data(head_list(list, false)), "David"));
BOOST_CHECK(size_list(list) == 5);
BOOST_CHECK(check_string(*(char **)temp, "David")); list_node_t *lnode = head_list(list, true);
BOOST_CHECK(check_string(get_string_data(lnode), "David"));
BOOST_CHECK(size_list(list) == 4); BOOST_CHECK(size_list(list) == 4);
// To free a node, free both the data and reference to data delete_node(list, lnode);
free_string(temp);
free(temp);
} }
BOOST_FIXTURE_TEST_CASE(test_tail_list, FixtureStrings) { BOOST_FIXTURE_TEST_CASE(test_tail_list, FixtureStrings) {
void *temp = tail_list(list); BOOST_CHECK(check_string(get_string_data(tail_list(list)), "Jimi"));
BOOST_CHECK(check_string(*(char **)temp, "Jimi"));
BOOST_CHECK(size_list(list) == 5); BOOST_CHECK(size_list(list) == 5);
free(temp);
} }
@@ -198,7 +194,8 @@ BOOST_AUTO_TEST_CASE(test_struct_list){
BOOST_CHECK(size_list(list) == 3); BOOST_CHECK(size_list(list) == 3);
//for_each_list(list, iterate_test_data); for_each_list(list, iterate_test_data);
list_node_t *lnode; list_node_t *lnode;
// Iterate over list while maintaining containment of abstraction // Iterate over list while maintaining containment of abstraction
@@ -207,6 +204,9 @@ BOOST_AUTO_TEST_CASE(test_struct_list){
printf("Found number: %i name: %s\n", test_data->num, test_data->name); printf("Found number: %i name: %s\n", test_data->num, test_data->name);
} }
lnode = head_list(list, true);
delete_node(list, lnode);
delete_list(list); delete_list(list);
} }