Description
1. Overview: Tries
We have seen in class that the trie data structure can be used to store strings. It provides for efficient
string insertion and lookup; insertion into a trie is O(k) (where k is the length of the string being
inserted), and searching for a string is an O(k) operation (worst-case). In a trie, a node does not store
the string it represents; rather, the edges taken to reach that node from the root indicate the string it
represents. Each node contains a flag (or count) variable that indicates whether the string represented
by that node has been inserted into the trie (or how many times it has been inserted). For example:
Figure 1:
This is a trie that codifies the words “and,” “cherry,” “chocolate,” “I,”
“like,” “really,” and “pie.” The strings “I,” “like,” and “pie” are
represented (or counted) twice. All other strings are counted only once.
0
0 0 2 0 0
0
0
2
0
2
0
1
h
d
p
i
e
i
l
i
k
e
c
0
n
0 0
e o
0 0
0 0
1 0
r
r
y
c
o
l
0
0
1
a
t
e
root
a
“pie” (2)
“like” (2)
“chocolate” (1)
“cherry” (1)
“and” (1)
0
0
0
r
e
a
0
0
1
l
l
y
“really” (1)
“i” (2)
1.1. TrieNode Struct (TriePrediction.h)
In this assignment, you will insert words from a corpus (that is, a body of text from an input file) into a
trie. The struct you will use for your trie nodes is as follows:
typedef struct TrieNode
{
// number of times this string occurs in the corpus
int count;
// 26 TrieNode pointers, one for each letter of the alphabet
struct TrieNode *children[26];
// the co-occurrence subtrie for this string
struct TrieNode *subtrie;
} TrieNode;
You must use this trie node struct, which is specified in TriePrediction.h, without any
modifications. You must #include the header file from TriePrediction.c like so:
#include “TriePrediction.h”
Notice that the trie node, because it only has 26 children, represents strings in a case insensitive way
(i.e., “apple” and “AppLE” are treated the same in this trie).
1.2. Subtries: Contextual Co-occurrence and Predictive Text
Words that appear together in the same sentence are said to “co-occur.” In this program, we’ll be
interested in contextual co-occurrence and predictive text – namely, given some word, w, what are all
the words that we see immediately after w in the sentences in some corpus? Consider, for example, the
following sentence:
I like cherry pie, and I really like chocolate pie.
In the example sentence, the word “I” is followed by the words “like” and “really”, the word “pie” is
followed by the word “and”, and so on.
To track co-occurrence, for each word w, we’ll place each word that directly follows w into the subtrie
of w. For example, if we place these terms (and their associated counts) into their own tries, those tries
will look like this:
(See following page for diagram.)
Figure 2:
Co-occurrence subtries for “I” (left) and “like” (right), based on the
sentence, “I like cherry pie, and I really like chocolate pie.”
In Figure 2, the trie on the left is what we will call the co-occurrence subtrie for the word “I,” based on
the example sentence above (I like cherry pie, and I really like chocolate pie). Its root should be stored
in the subtrie pointer field of the node marked “I” in the original trie diagram in Figure 1 (see page 2
above).
Similarly, the trie on the right in Figure 2 is the co-occurrence subtrie for the word “like.” Its root
should be stored in the subtrie pointer field of the node marked “like” in the original trie diagram in
Figure 1 (see page 2, above).
Within these subtries, all subtrie pointers should be initialized to NULL, because we will NOT produce
sub-subtries in this assignment!
0
0
0
0
1
l
i
k
e
root
0
0
e
0
0
0
a
l
l
r
1
y
0
0
0
0
0
o
c
o
l
root
0
0
e
0
0
1
r
r
y
c
0
h
0
0
1
a
t
e
2. Command Line Arguments
Your program will take two command line arguments, specifying two files to be read at runtime. The
first filename specifies a corpus that will be used to construct your trie and subtries. The second
filename specifies an input file with commands to be processed based on the contents of your trie. The
specifications for these files are described in the sections that follow. You can assume that we will
always specify valid corpus and input files when we run your program. For example:
./a.out corpus01.txt input01.txt
For a refresher on how to process command line arguments in C, please refer to the PDF for Program
#1 (Ohce).
3. Corpus File Format
The corpus file contains a series of sentences. Each line contains a single sentence with at least 1 word
and no more than 30 words. Each word contains fewer than 1024 characters, some of which may be
punctuation characters.
Each sentence will contain exactly one of the following punctuators, and it will occur at the very end of
the sentence: period (‘.’), exclamation point (‘!’), or question mark (‘?’). Other punctuators may occur
throughout the sentence, including (but not limited to) commas (‘,’), colons (‘:’), semicolons (‘;’),
apostrophes (‘’’ and ‘‘’), quotation marks (‘“’ and ‘”’), and so on.
All words will have at least one alphabetic character (‘a’ through ‘z’ or ‘A’ through ‘Z’). Words will not
contain any numeric characters (‘0’ through ‘9’). All punctuators should be stripped away from all
words before entering them into the trie. So, for example, the word “don’t” will be entered into the trie
as “dont”.
For example:
corpus01.txt:
I like cherry pie, and I really like chocolate pie.
corpus02.txt:
’tweren’t my fault, I swear!
And now we’re on to the second line of text.
4. Building the Tries (and Subtries)
First and foremost, each word from the corpus file should be inserted into your trie. If a word occurs
multiple times in the corpus, you should increment count variables accordingly. For example, the trie in
Figure 1 (see page 2, above) corresponds to the text given in the corpus01.txt file above.
For each sentence in the corpus, you must also update the co-occurrence subtrie for each word in that
sentence. The structure of the co-occurrence subtries is described above in Section 1.2, “Subtries:
Contextual Co-occurrence and Predictive Text.” If a string in the main trie is not followed by any other
words in the corpus, its subtrie pointer should be NULL.
5. Input File (Trie Processing Commands)
One of the required functions for this program needs to open and process an input file (the second
filename specified as a command line argument) that contains commands for you to process after
building your trie. Each line in this file will correspond to one of the following three commands:
Command Description
@ str n This is the text prediction command. When you encounter this command, you
should print the following sequence of (n + 1) words:
w0 w1 w2 … wn
In that sequence, w0 is str, and for 1 ≤ i ≤ n, wi is the word that most frequently
follows word wi-1 in the corpus. Note that the words are separated by spaces, and
there is never a space after the last word on one of these lines of output.
Furthermore, w0 is always capitalized in the output exactly as it was capitalized in
the command file, but words w1 through wn should appear in all lowercase.
If str does not appear in the trie, it should appear on this line of output by itself. If
some wi does not have any words in its subtrie, the sequence should terminate
prematurely. Again, this line of output should not have a trailing space at the end
of it, even if it terminates prematurely.
str If str is in the trie, print the string, followed by a printout of its subtrie contents,
using the output format shown in the sample output files for this program. Note
that when printing a subtrie, the words in the subtrie are preceded by hyphens.
If str is in the trie, but its subtrie is empty, you should print that string, followed
on the next line by “(EMPTY)”. If str is not in the trie at all, you should print that
string, followed on the next line by “(INVALID STRING)”.
! The character ‘!’ will appear on a line by itself. When you encounter this
command, you should print the trie using the output format shown in the sample
outputs for this program. (When printing the main trie, there are no hyphens
preceding the words on each line. See sample output files, or see the sample
output below on pgs. 7 and 8.)
Important note: In the commands listed above, str is always a string, and n is a non-negative integer.
Note that str is guaranteed to contain alphabetical characters only (‘A’ through ‘Z’ and ‘a’ through ‘z’).
Not counting the need for a null terminator (‘\0’), the length of str can range from 1 through 1023
characters (inclusively). So, with the null terminator, you might need up to 1024 characters to store str
as a char array when reading from the input file.
For more concrete examples of how these commands work or how your program’s output should be
formatted, see the attached input/output files and check out the function descriptions below in Section
8, “Function Requirements” (page 9).
6. Sample Input and Output Files
Consider, for example, the following corpus file and input (command) file:
corpus03.txt:
I like cherry pie and chocolate pie.
input03.txt:
!
chocolaTE
apricot
@ I 11
@ chocolate 1
@ persimmon 20
If passed to your program, those files should result in the following output:
output03.txt:
and (1)
cherry (1)
chocolate (1)
i (1)
like (1)
pie (2)
chocolaTE
– pie (1)
apricot
(INVALID STRING)
I like cherry pie and chocolate pie and chocolate pie and chocolate
chocolate pie
persimmon
The fun continues on the following page!
Note that a word might be in the trie but have an empty subtrie. Consider the following example:
corpus04.txt:
Spin straw to gold.
Spin all night long.
Spin spin spin.
Spindle.
input04.txt:
spin
spindle
nikstlitslepmur
output04.txt:
spin
– all (1)
– spin (2)
– straw (1)
spindle
(EMPTY)
nikstlitslepmur
(INVALID STRING)
You must follow the output format above precisely. Be sure to consult the included test case files for
further examples.
7. Trie Printing Functions (Included with Assignment!)
I have included some functions that will help you print the contents of your trie(s) in the required
format, because I think those functions are a bit too tricky for me to expect you to write them on your
own. See TriePrediction.c (attachment) for those functions. You’re welcome to use those functions
in the source file you submit, or you can write your own.
Also, studying and understanding those functions will serve as a launching point for you to write the
other functions required to get this program working.
The fun continues on the following page!
8. Function Requirements
In the source file you submit, TriePrediction.c, you must implement the following functions. You
may implement any auxiliary functions you need to make these work, as well. Please be sure the
spelling, capitalization, and return types of your functions match these prototypes exactly.
Note: I did not include any unit tests for processInputFile(), but I do intend to deploy unit tests for
that function when grading your assignment.
int main(int argc, char **argv);
Description: You must write a main() function for this program. It should do the following
five things: (1) capture the names of the corpus and input files (passed as command line
arguments), (2) call the buildTrie() function, (3) call the processInputFile() function, (4)
call the destroyTrie() function, and (5) return zero.
Returns: 0 (zero).
TrieNode *buildTrie(char *filename);
Description: filename is the name of a corpus text file to process. Open the file and create a
trie (including all its appropriate subtries) as described above.
Returns: The root of the new trie.
int processInputFile(TrieNode *root, char *filename);
Description: This function takes in the root of a trie and the name of an input file, and
processes that file according to the description above in Section 5, “Input File (Trie Processing
Commands).” While we will always specify valid filenames as command line arguments, we
might pass invalid filenames when unit testing. In the event that a bad filename is passed to this
function (i.e., the specified file does not exist), this function should simply return 1 without
producing any output.
Output: This function should produce output according to the specification described above in
Section 5, “Input File (Trie Processing Commands).” For additional details or clarification on
the output, please be sure to refer to the input/output files included with this assignment. Note
that this function should not produce any output if the input file does not exist.
Returns: If the specified input file does not exist, return 1. Otherwise, return 0.
TrieNode *destroyTrie(TrieNode *root);
Description: Free all dynamically allocated memory associated with this trie.
Returns: NULL.
TrieNode *getNode(TrieNode *root, char *str);
Description: Searches the trie for the specified string, str.
Returns: If the string is represented in the trie (with count ≥ 1), return a pointer to its terminal
node (the last node in the sequence, which represents that string). Otherwise, return NULL.
void getMostFrequentWord(TrieNode *root, char *str);
Description: Searches the trie for the most frequently occurring word and copies it into the
string variable passed to the function, str. If you are calling this function yourself, you should
create the str char array beforehand, and it should be (at least) long enough to hold the string
that will be written to it. (For this, you can use MAX_CHARACTERS_PER_WORD from
TriePrediction.h.) If we call this function manually when testing your code, we will ensure the
str char array is created ahead of time and that it is long enough to hold the longest string in
the trie. Note that there is no guarantee that str will contain the empty string when this function
is first called; the string might contain garbage data.
If there are multiple strings in the trie that are tied for the most frequently occurring, populate
str with the one that comes first in alphabetical order. If the trie is empty, set str to the empty
string (“”).
Hint: You might find it easier to write helper functions that you call from within this function.
Returns: Nothing. This is a void function.
int containsWord(TrieNode *root, char *str);
Description: Searches the trie for the specified string, str.
Note: You might find that you don’t need this function to build out the text prediction
functionality of your code, but you still need to implement it as part of this assignment.
Returns: If the string is represented in the trie (with count ≥ 1), return 1. Otherwise, return 0.
int prefixCount(TrieNode *root, char *str);
Description: Counts the number of strings in the trie (with count ≥ 1) that begin with the
specified string, str. Note that if the specified string itself is contained within the trie, that
string should be included in the count. If one of these strings occurs more than once, its entire
count should be added to the return value.
Note: You might find that you don’t need this function to build out the text prediction
functionality of your code, but you still need to implement it as part of this assignment.
Returns: The number of strings in the trie that begin with the specified string, str.
double difficultyRating(void);
Returns: A double indicating how difficult you found this assignment on a scale of 1.0
(ridiculously easy) through 5.0 (insanely difficult).
double hoursSpent(void);
Returns: An reasonable estimate (greater than zero) of the number of hours you spent on this
assignment.
9. Suggested Functions
These functions are not required, but I think they will simplify your task immensely if you implement
them properly and call them when processing your corpus/input files. Think of these function
descriptions as hints at how to proceed. If you want, you can even implement these functions with
different parameters, return types, and so on.
TrieNode *createTrieNode(void);
Description: Dynamically allocate space for a new TrieNode struct. Initialize all the struct
members appropriately.
Note: You should try to implement this yourself, but there’s a copy of this function in our notes
in Webcourses, should you really need it.
Returns: A pointer to the new node.
void insertString(TrieNode *root, char *str);
Description: Inserts the string str into the trie. Since it has no return value, it assumes the root
already exists (i.e., root is not NULL). If str is already represented in the trie, simply increment
its count member.
Note: You should try to implement this yourself, but there’s a copy of this function in our notes
in Webcourses, should you really need it.
Returns: Nothing. This is a void function.
void stripPunctuators(char *str);
Description: Takes a string, str, and removes all punctuation from the string. For example, if
str contains the string “Hello!” when the function is called, then str should contain the string
“Hello” when the function returns. When writing this function, you might find C’s built-in
isalpha() function (from ctype.h) to be helpful.
Returns: Nothing. This is a void function.
10.Special Requirement: Memory Leaks and Valgrind
Part of the credit for this assignment will be awarded based on your ability to implement the program
without any memory leaks. To test for memory leaks, you can use a program called valgrind, which is
installed on Eustis.
To run your program through valgrind at the command line, compile your code with the -g flag, and
then run valgrind, like so:
gcc TriePrediction.c -g
valgrind –leak-check=yes ./a.out corpus01.txt input01.txt
For help deciphering the output of valgrind, see: http://valgrind.org/docs/manual/quick-start.html
Note that if you do not use fclose() to explicitly close all open files before your program terminates,
valgrind might alert you that your program has a memory leak.
In the output of valgrind, the magic phrase you’re looking for to indicate that you have no memory
leaks is:
All heap blocks were freed – no leaks are possible
11.Test Cases and the test-all.sh Script
As always, we’ve included multiple test cases with this assignment to show some ways in which we
might test your code and to shed light on the expected functionality of your code. We’ve also included
a script, test-all.sh, that will compile and run all test cases for you. The script will also
automatically test for memory leaks using valgrind.
Using the test-all.sh script on Eustis is the safest, most sure-fire way to make sure your code is
working properly before submitting.
You can run the script on Eustis by placing it in a directory with TriePrediction.c,
TriePrediction.h, the sample_output directory, and all the test case files, and typing:
bash test-all.sh
Please note that these test cases are not comprehensive. You should also create your own test cases if
you want to test your code comprehensively. In creating your own test cases, you should always ask
yourself, “How could these functions be called in ways that don’t violate the function descriptions, but
which haven’t already been covered in the test cases included with the assignment?”
12.Compilation and Testing (Linux/Mac Command Line)
To compile at the command line:
gcc TriePrediction.c
By default, this will produce an executable file called a.out, which you can run by typing, e.g.:
./a.out corpus01.txt input01.txt
To redirect your output to a text file in Linux, just run the program as follows, which will create a file
called whatever.txt that contains the output from your program:
./a.out corpus01.txt input01.txt > whatever.txt
You can use diff to see whether your output matches ours exactly by typing, e.g.:
diff whatever.txt output01.txt
If the contents of whatever.txt and output01.txt are exactly the same, diff won’t have any output:
seansz@eustis:~$ diff whatever.txt output01.txt
seansz@eustis:~$ _
If the files differ, diff will spit out some information about the lines that aren’t the same. For example:
seansz@eustis:~$ diff whatever.txt output01.txt
11c11
< Like
—
> like
seansz@eustis:~$ _
12.1. Compiling and Running Unit Test Cases
We have also included unit tests that will call individual functions in your code to directly test their
functionality. To compile and run a unit test, first uncomment the #define main __hidden_main__
line in TriePrediction.h. Then, compile your source code like so:
gcc TriePrediction.c unit_test10.c UnitTestLauncher.c
To run unit tests in Code::Blocks, you will need to start a project and add all the appropriate files
(TriePrediction.c, TriePrediction.h, UnitTestLauncher.c, and unit_testXX.c.)
After running unit tests, before you can run standard test cases again, you will need to comment out the
#define main __hidden_main__ line in TriePrediction.h.
13. Deliverables
Submit a single source file, named TriePrediction.c, via Webcourses. The source file should contain
definitions for all the required function (listed above), as well as any auxiliary functions you need to
make them work. Don’t forget to #include “TriePrediction.h” in your source code.
Your program must work with the test-all.sh script, and it must be able to compile and run like so:
gcc TriePrediction.c
./a.out corpus01.txt input01.txt
Be sure to include your name and NID in a comment at the top of your source file.
14. Grading
The tentative scoring breakdown (not set in stone) for this programming assignment is:
90% correct output for test cases (including unit tests)
10% no memory leaks (passes valgrind tests)
Additional point deductions may be imposed for poor commenting and whitespace.
Special Restrictions: You must use tries in order to receive credit for this assignment. Also, please do
not use global variables in this program, do not make any system calls (e.g., system(“pause”)), and
do not write to any files. Violating any of these restrictions may result in a huge loss of points.
Note! Your program must be submitted via Webcourses, and it must compile and run on Eustis to
receive credit. Programs that do not compile will receive an automatic zero.
Your grade will be based primarily on your program’s ability to compile and produce the exact output
expected. Even minor deviations (such as capitalization, punctuation, or whitespace errors) in your
output will cause your program’s output to be marked as incorrect, resulting in severe point deductions.
The same is true of how you name your functions and their parameters. Please be sure to follow all
requirements carefully and test your program thoroughly.
For this program, we will also be unit testing your code. That means we will devise tests that determine
not only whether your program’s overall output is correct, but also whether certain required functions,
such as your buildTrie() function, do exactly what they are required to do. So, for example, if your
program produces correct output but your buildTrie() function is simply a skeleton that returns NULL
no matter what parameters you pass to it, or if it produces a totally funky, malformed trie, your program
will fail the unit tests.
Start early. Work hard. Ask questions. Good luck!