This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
haas:fall2014:data:projects:dlt0 [2014/11/16 16:47] – [In inc/tree.h] wedge | haas:fall2014:data:projects:dlt0 [2014/11/22 14:51] (current) – [Errata] wedge | ||
---|---|---|---|
Line 11: | Line 11: | ||
This section will document any updates applied to the project since original release: | This section will document any updates applied to the project since original release: | ||
- | * __revision | + | * __revision |
+ | * unit-addnode had a typo where it was expecting an empty tree to have a height of 0 instead of -1 (FIXED) | ||
+ | * unit-traverse had a number of typos, and saw some increased ruggedness: | ||
+ | * missing semi-colon on line 15 (FIXED) | ||
+ | * invalid variable reference (myList instead of tmpList) on/around line 52 (FIXED) | ||
+ | * extra checks to test invalid modes | ||
+ | * unit-set_mode has been deployed | ||
+ | * valid and invalid modes are checked | ||
+ | * unit-searchtree has been deployed | ||
+ | * NULL, empty, and non-empty tree searches of valid and invalid values | ||
+ | * verify-node.sh has been updated to show absolute totals | ||
+ | * verify-tree.sh has been updated to include new unit tests in its totals | ||
+ | * __revision 2__: new unit tests and verify scripts added (20141121) | ||
+ | * unit-balance has been deployed | ||
+ | * corresponding verify-balance.sh has been deployed | ||
+ | * the unit test for balance assumes a pivot based on the middle (if odd), or left-middle (if even) node | ||
+ | * unit-grabnode has been deployed | ||
+ | * corresponding verify-grabnode.sh has been deployed | ||
+ | * the grabnode unit test, like traverse, tests all 3 implementations in one unit test | ||
+ | * verify-tree.sh has been updated to include new unit tests | ||
+ | * __revision 3__: tweaks, comments, and new unit tests and verify script added (20141122) | ||
+ | * unit-grabnode may not have been working truly as intended. FIXED | ||
+ | * also added a specific test for grabbing from a single node tree | ||
+ | * performs 6 tests in total- NULL, empty, one-node, one node grabbed, two nodes grabbed, three nodes grabbed | ||
+ | * unit-addnode has added tests (looks like 35 in total when correctly implemented) | ||
+ | * unit-copytree has been deployed | ||
+ | * corresponding verify-copytree.sh has been deployed | ||
+ | * we perform 6 tests- NULL, empty, one-node, populated, populated and balanced, populated and node grabbed | ||
+ | * verify-tree.sh has been updated to include new unit test | ||
=====Objective===== | =====Objective===== | ||
Line 19: | Line 47: | ||
A **tree** is our first foray into the realm of non-linear data structures. While it still consists of nodes, and those nodes are linked together, they are not linked together in a way providing a contiguous succession as we have in a list. | A **tree** is our first foray into the realm of non-linear data structures. While it still consists of nodes, and those nodes are linked together, they are not linked together in a way providing a contiguous succession as we have in a list. | ||
- | The simplest tree is one which has a minimum of two connections- often these are called ' | + | Truth be told, the simplest tree is merely a list- the underlying data structure we've worked with all semester. But now, under this new light, we can start to look at a list as merely a procession of singled childed nodes. The more things are different, the more they are the same... |
+ | |||
+ | The simplest | ||
The premise behind a tree is that of a parent-child relationship. Each node can be a parent (of two child nodes), and each child can in turn be its own parent (of two child nodes). This creates a connection between relationships of nodes (ie a parent and its descendants), | The premise behind a tree is that of a parent-child relationship. Each node can be a parent (of two child nodes), and each child can in turn be its own parent (of two child nodes). This creates a connection between relationships of nodes (ie a parent and its descendants), | ||
Line 38: | Line 68: | ||
* a tree has a **root** which forms the basis or starting point of the structure; all nodes are reachable (and related through means of descendance) from the root. | * a tree has a **root** which forms the basis or starting point of the structure; all nodes are reachable (and related through means of descendance) from the root. | ||
* to put an item on the tree, we insert or **add** it (which you will have 3 separate opportunities to implement in this project!). When we add a node to the tree, it gets positioned in the appropriately available child link off some parent node. | * to put an item on the tree, we insert or **add** it (which you will have 3 separate opportunities to implement in this project!). When we add a node to the tree, it gets positioned in the appropriately available child link off some parent node. | ||
- | * to get an item off of the tree, we **get** it. Just as with a list, disconnecting a node from the structure can involve a lot of work to maintain connections and overall integrity of the data structure. | + | * to get an item off of the tree, we **grab** it (my term- often times it is referred to as deleting a node from the tree). Just as with a list, disconnecting a node from the structure can involve a lot of work to maintain connections and overall integrity of the data structure. |
+ | * an important action that will be taking place is the traversal of the tree- ie walking it in certain orders to achieve desired results (sorting/ | ||
As we will be placing nodes in the tree according to their stored values (lesser values go to the left, or previous; greater values go to the right, or next), our tree will naturally sort our data for us. | As we will be placing nodes in the tree according to their stored values (lesser values go to the left, or previous; greater values go to the right, or next), our tree will naturally sort our data for us. | ||
Line 81: | Line 112: | ||
#define _TREE_H | #define _TREE_H | ||
- | #include "node.h" | + | #include "list.h" |
- | # | + | # |
- | # | + | # |
- | # | + | # |
- | # | + | # |
- | # | + | # |
- | # | + | # |
+ | typedef struct tree Tree; | ||
struct tree { | struct tree { | ||
Node *root; | Node *root; | ||
- | char | + | char max_height; |
- | char | + | char height; |
- | Tree *(*add) (Tree *, Node * ); // addnode node (function pointer) | + | Tree *(*copy)(Tree * |
- | | + | |
- | | + | |
}; | }; | ||
- | typedef struct tree Tree; | ||
Tree *mktree | Tree *mktree | ||
Line 105: | Line 136: | ||
Tree *set_mode | Tree *set_mode | ||
- | Tree *addnode_s | + | Tree *addnode |
- | Tree *addnode_r | + | |
- | Tree *addnode_i | + | Tree *copytree_s(Tree * |
+ | Tree *copytree_r(Tree * | ||
+ | Tree *copytree_i(Tree * ); // copy/ | ||
+ | |||
+ | Node *searchtree(Tree *, char ); // find node in tree | ||
- | Node *findnode_s(Tree *, char ); // stack-based | + | Tree *grabnode_s(Tree *, Node **); // stack-based |
- | Node *findnode_r(Tree *, char ); // recursive | + | Tree *grabnode_r(Tree *, Node **); // recursive |
- | Node *findnode_i(Tree *, char ); // iterative | + | Tree *grabnode_i(Tree *, Node **); // iterative |
- | Tree *getnode_s | + | List *traverse_s(Tree *, char ); // stack-based |
- | Tree *getnode_r | + | List *traverse_r(Tree *, char ); // recursive |
- | Tree *getnode_i | + | List *traverse_i(Tree *, char ); // iterative |
- | Tree *cptree | ||
Tree *rmtree | Tree *rmtree | ||
- | Tree *balance | + | Tree *balance |
- | void show_tree (Tree *, char | ||
void print_tree(Tree * | void print_tree(Tree * | ||
Line 128: | Line 161: | ||
</ | </ | ||
- | ====list library==== | ||
- | One more enhancement is in store for our venerable **display()** function, and that is described below: | ||
- | ===display() enhancements=== | + | ====tree library==== |
- | You are also to implement an additional feature (resulting in 8 additional modes) to the **display()** function. | + | In **src/tree/**, you will find skeletons of the above prototyped functions, hollowed out in anticipation of being made operational. |
- | + | ||
- | This means you'll need to expand the current modulus divide by 8 to one of 16. | + | |
- | + | ||
- | The current modes are as follows: | + | |
- | + | ||
- | <code c> | + | |
- | // | + | |
- | // 1 display the list forward, with positional values | + | |
- | // 2 display the list backwards, no positional values | + | |
- | // 3 display the list backwards, with positional values | + | |
- | // 4 display the list forward, no positional values, in ASCII | + | |
- | // 5 display the list forward, with positional values, in ASCII | + | |
- | // 6 display the list backwards, no positional values, in ASCII | + | |
- | // 7 display the list backwards, with positional values, in ASCII | + | |
- | </ | + | |
- | + | ||
- | The new feature is to toggle the display of the individual arrows (and the terminating NULL value) and much of the natural spacing we've used so far. It should also ditch, when in ASCII mode, the quotes we have been placing around the ASCII characters. | + | |
- | + | ||
- | The idea here is, when combined with the ASCII representation of the list, **display()** can become a pretty useful function for displaying strings, which could come in handy. | + | |
- | + | ||
- | Also, when dealing with numeric values, we could make use of this in applications where we are handling the construction of a numeric value (such as a bignum). | + | |
- | + | ||
- | For positioned displays, this would seem to have less utility, but by having a space displayed before the position value, we can salvage the output nicely. | + | |
- | + | ||
- | So for each of the above 8 modes, you'll have a WITH_ARROWS mode, and then a WITHOUT_ARROWS mode (leaving us with a total of 16 possible combinations). | + | |
- | + | ||
- | For sane implementation purposes, for our current 8 (so modes 0-7), those will be considered to be WITH_ARROWS... whereas the remaining 8 combinations (8-15) will be WITHOUT_ARROWS. All other mode features will map in place (if you recognize the binary pattern I've been following you'll hopefully see how trivial this change (and the previous one) will be). | + | |
- | + | ||
- | Also, when not displaying arrows, no NULL values will be displayed at all; this means that in the event we are running **display()** on a NULL or empty list, nothing but a newline should be displayed. | + | |
- | + | ||
- | ==display() output examples based on mode== | + | |
- | Let's say we have a list with the following elements: | + | |
- | + | ||
- | start -> 51 -> 49 -> 51 -> 51 -> 55 -> NULL | + | |
- | + | ||
- | ==mode " | + | |
- | < | + | |
- | 5149515155 | + | |
- | </ | + | |
- | + | ||
- | ==mode " | + | |
- | < | + | |
- | [0]51 [1]49 [2]51 [3]51 [4]55 | + | |
- | </ | + | |
- | + | ||
- | ==mode " | + | |
- | < | + | |
- | 31337 | + | |
- | </ | + | |
- | + | ||
- | ==mode " | + | |
- | < | + | |
- | [0]3 [1]1 [2]3 [3]3 [4]7 | + | |
- | </ | + | |
- | + | ||
- | ====queue | + | |
- | In **src/queue/**, you will find skeletons of the above prototyped functions, hollowed out in anticipation of being made operational. | + | |
Figure out what is going on, the connections, | Figure out what is going on, the connections, | ||
- | Again, your queue is to utilize | + | Of particular focus in this project, in addition |
====Reference Implementation==== | ====Reference Implementation==== | ||
As the layers and complexities rise, narrowing down the source of errors becomes increasingly important. | As the layers and complexities rise, narrowing down the source of errors becomes increasingly important. | ||
- | If your stack push() isn't working, is it because of a problem | + | If your tree copytree_i() isn't working, is it because of a problem |
- | To aid you in your development efforts, you now have the ability to import | + | To aid you in your development efforts, you now have the ability to import functioning node, list, and stack implementations |
- | This also provides another opportunity for those who have fallen behind to stay current, as they can work on this project without having needed to successfully get full list functionality. | + | This also provides another opportunity for those who have fallen behind to stay current, as they can work on this project without having needed to successfully get full node or stack (and by extension- |
===Using the test reference implementation=== | ===Using the test reference implementation=== | ||
Line 214: | Line 187: | ||
</ | </ | ||
- | In order to make use of it, you'll need to run **make use-test-reference** from the base of your **dlq0** project directory, as follows: | + | In order to make use of it, you'll need to run **make use-test-reference** from the base of your **dlt0** project directory, as follows: |
<cli> | <cli> | ||
- | lab46: | + | lab46: |
... | ... | ||
- | NODE and LIST reference implementation in place, run ' | + | NODE, LIST, and STACK reference implementation in place, run ' |
- | lab46: | + | lab46: |
</ | </ | ||
Line 226: | Line 199: | ||
<cli> | <cli> | ||
- | lab46: | + | lab46: |
... | ... | ||
</ | </ | ||
- | **__Debugging__: | + | **__Debugging__: |
===Reverting back to using your code=== | ===Reverting back to using your code=== | ||
- | If you were trying out the reference implementation to verify | + | If you were trying out the reference implementation to verify |
<cli> | <cli> | ||
- | lab46: | + | lab46: |
- | Local node/list implementation restored, run 'make clean; make' to build everything. | + | Local node/list/ |
- | lab46: | + | lab46: |
</ | </ | ||
- | ====List Library unit tests==== | + | ====Node Library unit tests==== |
- | As a result of the required changes to **display()**, an updated unit test will be released (this will mean a different number of total tests for list than previously). | + | As a result of the addition of **other**, an updated |
+ | |||
+ | Be sure to run the various node unit tests and verification scripts to see which functions have fallen out of compliance with the node struct specification changes issued in this project. The **verify-node.sh** script can be especially useful in getting a big picture view of what work is needed. | ||
+ | ====Tree library unit tests==== | ||
+ | In **testing/ | ||
- | Be sure to run the various | + | * **unit-addnode.c** - unit test for the various |
- | ====Queue | + | * **unit-searchtree.c** - unit test for the various **searchtree()** library function |
- | In **testing/ | + | * **unit-grab.c** - unit test for **grabnode_*()** library functions |
+ | * **unit-set_mode.c** - unit test for **set_mode()** library function | ||
+ | * **unit-balance.c** - unit test for **balance()** library function | ||
+ | * **unit-treeops.c** | ||
+ | * **unit-traverse.c** - unit test for **traverse_*()** library functions | ||
+ | | ||
- | | + | **NOTE**: At the time of project release, |
- | * **unit-cpqueue.c** - unit test for **cpqueue()** library function | + | |
- | * **unit-rmqueue.c** - unit test for **rmqueue()** library function | + | |
- | * **unit-enqueue.c** - unit test for **enqueue()** library function | + | |
- | * **unit-dequeue.c** - unit test for **dequeue()** library function | + | |
- | * **unit-purge.c** | + | |
- | There are also corresponding **verify-FUNCTION.sh** scripts that will output a " | + | There are also corresponding **verify-FUNCTION.sh** scripts that will output a " |
- | These are complete runnable programs (when compiled, and linked against the queue library, which is all handled for you by the **Makefile** system in place). | + | These are complete runnable programs (when compiled, and linked against the tree library, which is all handled for you by the **Makefile** system in place). |
Of particular importance, I want you to take a close look at: | Of particular importance, I want you to take a close look at: | ||
Line 272: | Line 249: | ||
=====Expected Results===== | =====Expected Results===== | ||
- | To assist you in verifying a correct implementation, | + | To assist you in verifying a correct implementation, |
- | ====list library==== | + | ====node library==== |
- | Here is what you should get for list: | + | Here is what you should get for node: |
<cli> | <cli> | ||
- | lab46: | + | lab46: |
==================================================== | ==================================================== | ||
- | = Verifying Doubly-Linked | + | = Verifying Doubly-Linked |
==================================================== | ==================================================== | ||
- | [mklist] Total: | + | [mknode] Total: |
- | [cplist] Total: | + | [cpnode] Total: |
- | [rmlist] Total: | + | [rmnode] Total: |
- | [append] Total: | + | |
- | [insert] Total: | + | |
- | [obtain] Total: | + | |
- | | + | |
- | [findnode] Total: | + | |
- | [sortlist] Total: | + | |
- | [swapnode] Total: | + | |
==================================================== | ==================================================== | ||
- | | + | |
==================================================== | ==================================================== | ||
- | lab46: | + | lab46: |
</ | </ | ||
- | With the added feature to **display()**, we have 10 additional combinations to test, plus I added in 4 explicit tests of invalid modes, so this will bump up the total number of list tests by 10+4, going from the dll0 total of 102 to the dls0 total of 140 to the dlq0 total of 154. | + | With the addition of the **other** pointer, we have a few additional combinations to test. |
- | But aside from this change to **display()**, | ||
- | ====queue library==== | + | ====tree library==== |
- | Here is what you should get for queue: | + | Here is what you should get for tree: |
<cli> | <cli> | ||
- | lab46: | + | lab46: |
- | =================================================== | + | coming soon |
- | = | + | lab46: |
- | =================================================== | + | |
- | [mkqueue] Total: | + | |
- | [cpqueue] Total: | + | |
- | [rmqueue] Total: | + | |
- | [purge] Total: | + | |
- | [enqueue] Total: | + | |
- | [dequeue] Total: | + | |
- | =================================================== | + | |
- | [RESULTS] Total: | + | |
- | =================================================== | + | |
- | lab46: | + | |
</ | </ | ||
Line 327: | Line 285: | ||
* Project must be submit on time, by the posted deadline. | * Project must be submit on time, by the posted deadline. | ||
- | * Late submissions will lose 25% credit per day, with the submission window closing on the 4th day following the deadline. | + | * As this project is due at the very end of the semester, there is no late window. It is either submitted, or it is not. |
* All code must compile cleanly (no warnings or errors) | * All code must compile cleanly (no warnings or errors) | ||
* all requested functions must be implemented in the related library | * all requested functions must be implemented in the related library |