Skip to content

Latest commit

 

History

History
164 lines (132 loc) · 7.02 KB

README.md

File metadata and controls

164 lines (132 loc) · 7.02 KB

Lab Project 1: Building a Tokenizer

A related demo program is in directory "sortArgDemo". You are encouraged to thoroughly understand it before attempting this project.

All work should be your own work. Please see the course website and syllabus for full details on the course's policy on academic honesty.

Prototype source code for this project is in a directory named src. You should add and modify source code within that directory, and those changes should be pushed to your private Github repository prior to the deadline. Your project should contain:

  • All .c and .h source files
  • A Makefile whose default production builds the project, and whose clean production eliminates any files created when the project is built.
  • A brief README documenting any non-obvious aspects of your program. For example, you may want to describe how it works, how to run it, and any bugs or idiosyncracies the user ought to know about.

Learning outcomes

Students who complete this lab will demonstrate ability to

  • Use command-line tools (cp, mv, rm, mkdir, cd, emacs, cc, make, git) for program development under bash
  • Develop and debug C programs that include
    • Pointer arithmetic
    • Dynamic memory allocation & free (malloc/calloc/free)
    • Multiple source files (and headers with declarations)
    • A pointer-based data structure (linked list of structs)
  • Design a comprehensible small program in the C language.
  • You may refer to the syllabus, course info, lectures, and assignment for an in depth explanation of the grading criteria and expectations.

Grading

At a technical interview where your project will be examined and discussed, your mastery of the following will be assessed

  • Proper use of development tools (e.g. bash, emacs, make, compiler, and git)
  • Clear and concise written communication
  • Code Hygiene: consistent, expressive, and efficient
    • variable, function, and struct names
    • comments
    • modularization (into functions and files) including header files
    • storage management (variable scope & lifetime, allocation and freeing of dynamically allocated memory)
    • consistent and language-appropriate coding style
      • indentation
      • use of capitalization and whitespace (e.g. char *word_end(char *p) rather than char* wordEnd (char* p))
    • algorithms and data structures
    • repository content (all necessary files are included, no unnecessary files such as #tokenizer.h# and tokenizer.o)
    • repository log messages (we encourage you to examine ours)
  • Correctness
  • Timely completion

Overview

Your objective for the first lab project is to build a tokenizer that maintains a history. Tokenizing is the process of dividing a string into tokens by some delimiter (e.g. the string library's split() function). Consider tokenizing the string “The cake is a lie!” using the space character. This would result in: [“The”, “cake”, “is”, “a”, “lie!”].

Lastly, your tokenizer should maintain a history. A good example of this is the bash command, history; or, something similar to below:

  1. The cake is a lie!
  2. But it was delicious.
  3. True.

A user should be able to view and recall a certain history item by using a command similar to ’!2’. This would then use the string “But it was delicious.” as input and tokenize.

Building a Simple UI

The first milestone for your project is to complete a simple functioning user interface (which means it does not have to be anything fancy). It should have an indicator on the left to notify the user that the program is ready to receive input. For example:

$ this would work

> or even this

At this point of the project, your program should just echo the input back to the user.

$ my fancy interface thing
my fancy interface thing

Working with Strings

Inside this repository there are two header files, tokenizer.h and history.h, which declare functions your project will define. You should explore the contents of each file so that you are familiar with the instructions to follow.

The second milestone of your project is to define several of the functions declared and described (but not defined) in tokenizer.h:

  1. int space_char(char c); // true if c is a tab or space, and not zero
  2. int non_space_char(char c); // true if c not a tab or space, and not zero
  3. char *word_start(char *str); // * to first non-space char in first word in s
  4. char *word_end(char *str); // * to for space char after end of word
  5. int count_words(char *str); // the number of words in the string

The declarations of these functions in tokenizer.h includes a more comprehensive description of each function's semantics. We strongly encourage you to read these descriptions carefully, and to include them with your function declarations.

Each function definition should appropriately use the arguments and return the correct type declared in the method declaration. Your code should include tokenizer.h and not modify the the declarations it contains.

Lastly, you are not allowed to use any libraries except stdio.h and stdlib.h in your source code. You must implement all string functions yourself

Dealing with Memory Allocation

The third milestone of your project will require you to allocate and free memory. You must define and test the following functions declared in tokenizer.h.

  1. char *copy_str(char *src, short len); // return len characters from src
  2. void print_tokens(char **toks); // print the tokens pointed to by toks
  3. char *get_token(char **toks, int id); // return the idth token from toks
  4. void free_tokens(char **toks); // free all memory associated with toks

To test your functions, you should hardcode the allocation of an array of strings (char**).

Putting it All Together

Now that you have all the pieces for the tokenizer defined, the last part is to put them together and define the tokenizer:

char **tokenize(char *s);

As explained in the Overview, the tokenize() method should accept a string of characters as input and return an array of tokens. It is important to understand that you must determine the number of tokens needed prior to allocating space for the array, and similarly the size of each token before allocating space for the token.

Maintaining History

The final milestone of the project is to build the history component. The history can be represented as a linked list data structure, with each node containing:

  1. an id based on its place in the list,
  2. the string entered by user,
  3. and, a pointer to the next node.

Inside the history.h header file you will find two struct declarations representing this notion.

Overall, a user should be able to:

  1. view the history of strings entered by the user
  2. and recall a certain history item by using the command !3 (i.e. an exclamation directly followed by the sequence number).

You should define each function from the history.h header file in a separate history.c file and then incorporate the history with your tokenizer and interface.