forked from aaronbloomfield/pdr
-
Notifications
You must be signed in to change notification settings - Fork 228
/
index.html
446 lines (446 loc) · 22.3 KB
/
index.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml" lang="" xml:lang="">
<head>
<meta charset="utf-8" />
<meta name="generator" content="pandoc" />
<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" />
<title>PDR: Laboratory 10: Huffman Coding</title>
<style>
code{white-space: pre-wrap;}
span.smallcaps{font-variant: small-caps;}
span.underline{text-decoration: underline;}
div.column{display: inline-block; vertical-align: top; width: 50%;}
div.hanging-indent{margin-left: 1.5em; text-indent: -1.5em;}
ul.task-list{list-style: none;}
.display.math{display: block; text-align: center; margin: 0.5rem auto;}
</style>
<link rel="stylesheet" href="../../markdown.css" />
<!--[if lt IE 9]>
<script src="//cdnjs.cloudflare.com/ajax/libs/html5shiv/3.7.3/html5shiv-printshiv.min.js"></script>
<![endif]-->
</head>
<body>
<h1 id="pdr-laboratory-10-huffman-coding">PDR: Laboratory 10: Huffman
Coding</h1>
<p><a href="../index.html">Go up to the Labs table of contents
page</a></p>
<h3 id="objective">Objective</h3>
<ol type="1">
<li>To become familiar with prefix codes</li>
<li>To implement a useful application using a variety of data
structures</li>
<li>To analyze the efficiency of your implementation</li>
</ol>
<h3 id="background">Background</h3>
<p>In lecture we discussed Huffman coding and the construction of prefix
code trees. We have also covered a variety of data structures this
semester: lists, trees, hash tables, and heaps. Several of these may be
useful for this lab. In addition, in this lab you are required to think
about the underlying representation and efficiency of these
structures.</p>
<h3 id="tutorial">Tutorial</h3>
<p>There is no tutorial for this lab.</p>
<h3 id="recommended-readings">Recommended Readings</h3>
<ul>
<li>The <a href="../../slides/10-heaps-huffman.html">Heaps and Huffman
slide set</a> are available in the repository, and that covers Huffman
compression and decompression.</li>
<li>Code for binary heaps is also available from that slide set: <a
href="../../slides/code/10-heaps-huffman/binary_heap.cpp.html">binary_heap.cpp</a>
(<a href="../../slides/code/10-heaps-huffman/binary_heap.cpp">src</a>),
<a
href="../../slides/code/10-heaps-huffman/binary_heap.h.html">binary_heap.h</a>
(<a href="../../slides/code/10-heaps-huffman/binary_heap.h">src</a>), <a
href="../../slides/code/10-heaps-huffman/heap-test.cpp.html">heap-test.cpp</a>
(<a
href="../../slides/code/10-heaps-huffman/heap-test.cpp">src</a>).</li>
</ul>
<h2 id="lab-procedure">Lab Procedure</h2>
<h3 id="pre-lab">Pre-lab</h3>
<ol type="1">
<li>Create a heap</li>
<li>Implement the Huffman encoding (compression) routine using your
heap</li>
<li>Files to download: <a href="fileio.cpp.html">fileio.cpp</a> (<a
href="fileio.cpp">src</a>), and the example files (in the
labs/lab10/examples/ directory), or as one <a
href="examples.zip">examples.zip</a> file)</li>
<li>Files to submit: heap.cpp, heap.h, huffmanenc.cpp, Makefile (you can
submit additional .cpp/.h files, if needed, as long as it compiles with
<code>make</code>)</li>
</ol>
<h3 id="in-lab">In-lab</h3>
<p>For spring 2022, there is an extension to the in-lab due date; see
the <a href="../../uva/daily-announcements.html#/">daily
announcements</a> for details.</p>
<ol type="1">
<li>Implement the Huffman decoding (decompression) routine discussed in
the in-lab section</li>
<li>Files to download: <a
href="inlab-skeleton.cpp.html">inlab-skeleton.cpp</a> (<a
href="inlab-skeleton.cpp">src</a>), and your pre-lab files</li>
<li>Files to submit: huffmandec.cpp, Makefile (you can submit additional
.cpp/.h files, if needed, as long as it compiles with
<code>make</code>)</li>
</ol>
<h3 id="post-lab">Post-lab</h3>
<p>The post-lab is cancelled for the spring 2022 semester.</p>
<ol type="1">
<li><del>Analyze the time and space complexity of your encoding and
decoding code</del></li>
<li><del>Create a Makefile to compile both the encoding and decoding
portions of your Huffman routine in one step</del></li>
<li><del>Files to submit: Makefile, all necessary source code
files</del></li>
</ol>
<hr />
<h2 id="huffman-encoding-and-decoding">Huffman Encoding and
Decoding</h2>
<p>There are a few points that apply to the Huffman implementation as a
whole that are important to keep in mind.</p>
<h3 id="requirements">Requirements</h3>
<p>Your Huffman implementation should:</p>
<ul>
<li>Case-sensitively encode and decode <em>only</em> printable ASCII
characters
<ul>
<li>Newlines, tabs, etc. are not printable, while spaces are</li>
<li>The lecture notes describe <a
href="../../slides/10-heaps-huffman.html#asciiset">which characters are
to be encoded</a></li>
</ul></li>
<li>Read and write ASCII characters rather than individual bits
<ul>
<li>Yes, this means that the actual size of our compressed file will be
larger than our original file (terrific, you “compressed” the character
<code>t</code> into the five characters <code>01011</code>), but it will
simplify the lab considerably</li>
</ul></li>
<li>Use a heap (priority queue) data structure
<ul>
<li>You may NOT use the STL <code>priority_queue</code></li>
<li>You may use the heap code from the slides as long as you cite
it</li>
</ul></li>
</ul>
<h3 id="efficiency">Efficiency</h3>
<p>You will need to select several different data structures to
implement Huffman compression and decompression. Don’t get more
complicated than is necessary, but do keep efficiency in mind. Consider
the following questions:</p>
<ul>
<li>What will be the most common operations required on each data
structure?</li>
<li>How much time and space will be required?</li>
</ul>
<p>Use the answers to these questions to guide your selection. Your
solution will be judged slightly on how efficient it is (both in terms
of time and space). Very inefficient solutions will lose a few points.
Be sure you have a good explanation for your implementation choices:</p>
<ul>
<li>Do you predict better data locality?<br />
</li>
<li>Does a purely Big-Theta analysis not tell the whole story?</li>
</ul>
<p>Whatever your implementation, you should be able to accurately
describe its worst case Big-Theta performance both in running time and
space (memory) usage.</p>
<p>One thing we have not dealt with in previous labs is serializing
various data structures (writing them to a file or standard output, and
reading them back in). You will need to do this with your prefix code
tree. The fact that you must read this data structure from a file and
write to standard output may affect how you choose to represent it in
your program. Keep in mind that the format for how to write it to a file
is fixed, as described in the pre-lab section. You will need to
re-create your Huffman coding tree from the first part of that file
format.</p>
<hr />
<h2 id="pre-lab-1">Pre-lab</h2>
<p>For the pre-lab, you will implement, in huffmanenc.cpp, the Huffman
encoding algorithm using a binary heap that you implement in
heap.cpp/h.</p>
<p>The basic steps for compression are:</p>
<ol type="1">
<li>Read the source file and determine the frequencies of the characters
in the file</li>
<li>Store the character frequencies in a heap (priority queue)</li>
<li>Build a tree of prefix codes (a Huffman code) that determines the
unique bit codes for each character</li>
<li>Re-read the source file and for each character read, write its
prefix code to the output, following the file format described
below</li>
</ol>
<p>Your program must take in a single command-line parameter, which is
the name of the file whose contents will be encoded. We have some sample
plain text and encoded text files (in the labs/lab10/examples/
directory) – a description of these files is in the in-lab section.</p>
<p>Your program should output to the standard output
(i.e. <code>cout</code>) the exact file format described below, and
nothing else.</p>
<p>As part of the file format described below, your program will need to
print out the compression ratio and the Huffman tree cost. The
compression ratio is defined, in bits, as: (size of the original
file)/(size of compressed file). You should exclude the size of the
prefix code tree in the compression ratio, and assume that the 0’s and
1’s you generated for the compressed file count as one bit each (rather
than an 8-bit character). The Huffman tree cost is the same as described
in lecture.</p>
<p>To read in the input file character by character, see the <a
href="fileio.cpp.html">fileio.cpp</a> (<a href="fileio.cpp">src</a>)
file.</p>
<h3 id="file-format">File Format</h3>
<p>In an effort to simplify both the in-lab implementation and the
grading, there is a very strict file format for a Huffman encoded
message for this lab. This allows the two parts to be developed,
implemented, and tested separately. Although real Huffman encoding uses
bits, we will write these bits to a file using the characters
<code>0</code> and <code>1</code>, as that will make it easier to check
and debug our code.</p>
<p>There are three sections to an encoded file, each separated by a
specific separator.</p>
<h4 id="section-1">Section 1</h4>
<p>The first section of the file are the ASCII characters that are
encoded, followed by their bit encoding. Only one such encoding per
line, with no blank lines allowed. The format for a line is the ASCII
character, a single space, and then the <code>1</code> and
<code>0</code> characters that are the encoding for that character,
followed by a newline. The order of the lines does not matter. If the
character being written is the space character itself, you should write
<code>space</code> instead of ” “; thus a line for the space character
might look like: <code>space 101010</code>. Keep in mind that the first
character on this type of line cannot be a newline, tab, or a
non-printable character. No line in this part can be more than 256
characters. You can safely assume that the data provided will be a valid
Huffman coding tree (i.e. there won’t be an internal node with one
child, etc.).</p>
<p>Following that is a separator line, and is a single line containing
40 dashes and no spaces.</p>
<h4 id="section-2">Section 2</h4>
<p>The second section is the encoded message, using the characters
<code>0</code> and <code>1</code>. You may optionally separate each
encoded character by any form of whitespace. For example, to help
debugging, you might want to separate each encoded letter by a space and
place each encoded word on its own line.</p>
<p>The next line is a separator, and is also a single line containing 40
dashes and no spaces.</p>
<h4 id="section-3">Section 3</h4>
<p>The last section of the file displays several stats about the
compression. You should output the total number of symbols encoded, how
many unique symbols are used, the number of bits in the original and
compressed files, the compression ratio to five decimal places, and the
cost of the huffman tree to five decimal places. The output format
should look exactly as shown below. The information is required as we
will be grading your submissions with these results. All floating point
results must be reported to <strong>5 decimal places</strong>.</p>
<p>The following is the Huffman file format for the example in the slide
set that has the characers ‘a’, ‘b’, ‘c’, and ‘d’.</p>
<pre><code>a 0
b 100
c 101
d 11
----------------------------------------
11 100 0 101 0 0 11
----------------------------------------
There are a total of 7 symbols that are encoded.
There are 4 distinct symbols used.
There were 56 bits in the original file.
There were 13 bits in the compressed file.
This gives a compression ratio of 4.30769.
The cost of the Huffman tree is 1.85714 bits per character.</code></pre>
<p>The Huffman tree that this forms is the same as the one shown in <a
href="../../slides/10-heaps-huffman.html#/lab10tree">the slide set</a>),
and is duplicated below.</p>
<p><img src="prelab-tree.png" /></p>
<p>Note that your encoding does not have to exactly match – in
particular, the bits that your program uses to encode it will depend on
the implementation of your heap. So while your bits can be off, the
number of bits for each character should <strong>NOT</strong> be
different than the examples given. For example, in the output above, if
<em>b</em> was given the prefix code <em>101</em> and <em>c</em> was
given <em>100</em> (i.e., <em>b</em> and <em>c</em> have their codes
swapped), then that is ok. The statistics in section 3 of the output
will be the same.</p>
<h3 id="hints">Hints</h3>
<p>A few hints from experience with this in previous semesters.</p>
<h4 id="take-the-time-to-understand-the-code">Take the time to
understand the code</h4>
<p>Just copying the supplied code verbatim will not help you, as you
need to <em>understand</em> what is going on – the code does not work
“out of the box.” Either make sure you understand it, or write your own
code.</p>
<h4 id="need-a-pointer-or-two">Need a pointer or two?</h4>
<p>It will be <strong>far</strong> easier to put HuffmanNode pointers in
your heap, rather than HuffmanNode objects (or whatever your Huffman
tree nodes are called). If you put the actual objects in, then you are
going to run into problems with scoping issues, inadvertent calls to
<code>operator=()</code>, etc. Trust us, stick with the pointers.</p>
<h4 id="okay-but-actually">Okay but actually</h4>
<p>You will need some secondary data structures in order to hold all the
information necessary for compression. You’ll need some way of holding
the frequencies of each character, as well as each character’s prefix
codes – what would be efficient data structures to store and retrieve
that information?</p>
<h4 id="getting-the-prefix-codes">Getting the prefix codes</h4>
<p>The best way to do this is recursively. Once you have your Huffman
tree, you can recurse down both sides of your tree to grab every prefix
code and print them out.</p>
<hr />
<h2 id="in-lab-1">In-lab</h2>
<p>For the in-lab, you will implement, in huffmandec.cpp, the Huffman
decoding algorithm.</p>
<p>The basic steps for decompression are:</p>
<ol type="1">
<li>Read in the prefix code structure from the encoded file</li>
<li>Re-create the Huffman tree from the code structure read in from the
file</li>
<li>Read in one bit at a time from the encoded file and move through the
prefix code tree until a leaf node is reached</li>
<li>Output the character stored at the leaf node</li>
<li>Repeat the last two steps until the encoded file has been entirely
decoded</li>
</ol>
<p>First, make sure that your code can read in encoded files – you can
download the <a href="inlab-skeleton.cpp">inlab-skeleton.cpp</a> file,
which can properly read in the encoded input files.</p>
<p>Next, create the Huffman coding tree from the prefix codes you read
in from the input file. Not creating a Huffman tree from the file will
result in zero credit for the in-lab. The whole point of this part is to
create the tree!</p>
<p>Lastly, read in the second part of the file, traverse your Huffman
tree, and output a character whenever you reach a leaf node. For this
lab, you must only print out the decoded message and nothing else.</p>
<p>We provide a number of sample files for you to test your code with. A
brief description of each is described here. The “normal” files are the
English input. The “encoded” files are the Huffman encoded files,
following the file format described above. Except where indicated, the
second section of each encoded file (the digits <code>0</code> and
<code>1</code>) has a space inserted between each letter from the
original file, so that you can see which letter is encoded as which
bitcode.</p>
<ul>
<li><a href="examples/normal1.txt">normal1.txt</a> / <a
href="examples/encoded1.txt">encoded1.txt</a>: This is the first example
from the lecture slides (<code>dbacaad</code>). The Huffman tree can be
viewed <a href="prelab-tree.png">here</a>.</li>
<li><a href="examples/normal2.txt">normal2.txt</a> / <a
href="examples/encoded2.txt">encoded2.txt</a>: This is the second
example from the lecture slides, in the “Huffman Encoding” section. This
is the example that we built up the Huffman tree from. The Huffman tree
can be viewed <a href="inlab-tree-2.png">here</a>.</li>
<li><a href="examples/normal3.txt">normal3.txt</a> / <a
href="examples/encoded3.txt">encoded3.txt</a>: This is a paragraph from
<a href="https://en.wikipedia.org/wiki/Gadsby_%28novel%29">Gadsby</a>,
which is a novel that does not ever use the letter ‘e’.</li>
<li><a href="examples/normal4.txt">normal4.txt</a> / <a
href="examples/encoded4.txt">encoded4.txt</a>: The first paragraph from
a <a
href="https://www.cavalierdaily.com/article/2007/11/rolls-royce-partners-with-va-universities">front
page story in the 27 November 2007 edition of the Cavalier Daily</a>.
<ul>
<li><a href="examples/encoded4a.txt">encoded4a.txt</a>: This is the same
encoding as <a href="examples/encoded4.txt">encoded4.txt</a>, but with
all spaces in the second section of the file removed, so that it’s just
a very long string of <code>0</code>s and <code>1</code>s.</li>
</ul></li>
</ul>
<h3 id="hints-1">Hints</h3>
<h4 id="more-data-structures">More data structures</h4>
<p>Like the pre-lab, you’ll need some way to store the prefix codes for
each character. Unlike the pre-lab, though, you don’t need to deal with
frequencies anymore. The prefix codes is enough to generate the Huffman
tree, which you can then use to decode the input file.</p>
<h4 id="creating-the-huffman-tree">Creating the Huffman tree</h4>
<p>As you are (recursively) creating each node in the tree, you know the
<em>prefix</em> code to get to that node (remember that following a left
child pointer generates a <code>0</code>, and following a right child
pointer generates a <code>1</code>). If that prefix is one of the listed
bitcodes in the first part of the file, then we know we are at a leaf
(remember that all characters in a Huffman tree are leaves). Otherwise,
we are dealing with an internal node and you will have to create left
and right child nodes, calling yourself recursively on each one. Keep in
mind that when you call yourself recursively on the left child, you have
to add <code>0</code> to the end of the prefix; likewise, you have to
add <code>1</code> to the prefix for the right child. This algorithm
will require you to search through the bit codes that were read in from
the first part of the file. Also keep in mind that the size of the input
here (the number of characters) is very small (only 80 or so) – which
means that if you choose a linear time complexity data structure
(vector, for example), your code will run just fine.</p>
<hr />
<h2 id="post-lab-1">Post-lab</h2>
<p>The post-lab is cancelled for the spring 2021 semester.</p>
<p><del>There are two parts to this post-lab: the time and space
complexity analysis and submitting all your (working) code
again.</del></p>
<h3 id="time-and-space-complexity"><del>Time and Space
Complexity</del></h3>
<p><del>For the post-lab, we want you to think about the time and space
complexity analysis of your compression and decompression code. You
aren’t required to submit anything for this part, but you should think
carefully about the following points, as you may need to know them on
exams. See below for a discussion about the space/time
complexity.</del></p>
<ul>
<li><del>Think about your implmentation: what data structures did you
use in your implementation and <em>why</em> did you selected
them.</del></li>
<li><del>Analyze the efficiency of <em>all steps</em> in Huffman
encoding/decoding. For each of the steps of compression and
decompression (see “Huffman Encoding and Decoding”), what is:</del>
<ul>
<li><del>The worst case running time of your implementation</del></li>
<li><del>The worst case space complexity of your
implementation</del></li>
</ul></li>
</ul>
<p><del>Worst case running time – for this be sure to consider all steps
of the compression and decompression. You can leave off the cost of
calculating the compression ratio, printing the cost of the tree, and
printing a listing of the bit code for each character that was asked for
in the pre-lab. Refer to the list of steps given earlier in the
lab.</del></p>
<p><del>Space complexity – for this, you should calculate the number of
bytes that are used by each data structure in your implementation. The
easiest way to do this is to step through your code, just as you have
done for the worst case running time, and make a note each time you use
a new data structure. You do not need to take into account scalar
variables (loop counters, other singleton variables), focus on the data
structures whose size depends on values such as the total number of
characters and the total number of unique characters, and use those
values in your answer.</del></p>
<p><del>Again, you aren’t required to submit anything for this section,
but you should use these points as a way to reflect on the work you have
done for the pre-lab and in-lab.</del></p>
<h3 id="bringing-it-all-together"><del>Bringing it all
together</del></h3>
<p><del>The purpose of this part of the post-lab is to clean up your
code from the pre-lab and in-lab, and submit all of it together. If your
pre-lab and in-lab code work properly, then there is no futher clean-up
to do; however, you must still submit the files along with a
<em>new</em> Makefile.</del></p>
<p><del>When we run <code>make</code>, the code should be compiled into
two executables: <code>encoder</code> and <code>decoder</code>, which
are the pre-lab and in-lab code bases, respectively. Unlike the pre-lab
and in-lab, you should <strong><em>NOT</em></strong> name your
executables <code>a.out</code>! After compiling your code with
<code>make</code>, we will test it as such:</del></p>
<pre><code>./encoder testfile.txt > encoded.txt
./decoder encoded.txt > output.txt
diff testfile.txt output.txt</code></pre>
<p><del>This encodes a sample text file, then decodes it. Both the
original file (<code>testfile.txt</code>) and the final file
(<code>output.txt</code>) should be the same, which is what the
<code>diff</code> command checks. If there are no differences between
the two files, then <code>diff</code> will not print any
output.</del></p>
<h3 id="hints-2"><del>Hints</del></h3>
<h4 id="two-executables"><del>Two executables???</del></h4>
<p><del>Up until now, all of our Makefiles have generated a single
<code>a.out</code> from the inputs given through <code>OBJECTS</code>.
Now that we have two separate executables we need to create, you may
need some way to differentiate between the object files for each
one…</del></p>
<h4 id="naming-the-executables"><del>Naming the executables</del></h4>
<p><del>Remember the <code>-o</code> flag for clang++? :)</del></p>
</body>
</html>