辅导CS2201留学生、讲解C++ queue ADT

- 首页 >> C/C++编程


Fall 2019

Assignment No. 6

Purpose:

This project is meant to give you experience in creating and using a queue ADT. In particular, you will implement a queue

ADT that will hold doubles. Your queue implementation will then be used to simulate the plucking of a guitar string

using the Karplus-Strong algorithm. This algorithm played a seminal role in the emergence of physically

modeled sound synthesis (where a physical description of a musical instrument is used to synthesize sound

electronically). This assignment will build upon our knowledge of sound files that we learned in the previous

assignment.

The Assignment:

You will create a program that reads a text file containing information on what notes are to be generated and when they

are to be generated, and from that create a sound file in the .dat format (the same .dat format as the previous assignment).

You have been provided with a starter file for the main program, GuitarHero.cpp that does some initial file

processing for you. Your task will be to define a queue that holds double values (DblQueue.h & DblQueue.cpp),

and to define a class that represents a guitar string (GuitarString.h & GuitarString.cpp) which uses the

queue. Finally you will complete the main program to generate a correct .dat file.

You will be provided starter files for DblQueue.h & GuitarString.h, which define the public interfaces of the

classes. You need to complete DblQueue.h and create DblQueue.cpp to implement an unbounded queue. Your

code should use a singly linked list as the basis for your implementation. You should test your queue implementation fully

before proceeding (use DblQueueTest.cpp). Once you complete the queue implementation, you will use it to

complete GuitarString.h and create GuitarString.cpp, a class that represents a guitar string. You should test

your GuitarString implementation fully before proceeding (use GuitarStringTest.cpp). The full API for both

classes is specified below.

After completing & testing the DblQueue and GuitarString classes, you will complete the code in

GuitarHero.cpp. The program reads data from a file which indicates which strings are to be plucked and when, and it

generates a data file for the sox program.

Simulating the Plucking of a Guitar String:

[Note: This is background information, so don’t get bogged down in it. Implementation details begin in the next section

titled “Functional Specifications”.] When a guitar string is plucked, the string vibrates to create sound. The length of the

string determines its fundamental frequency of vibration. We model a guitar string by sampling its displacement (a real

number between -½ and +½) at N equally spaced points (in time), where N equals the sampling rate (44,100 times per

second) divided by the fundamental frequency (rounded to the nearest integer).

• Plucking the string. The excitation of the string can contain energy at any frequency. We simulate the excitation by

filling a buffer (a queue in our case) with white noise: set each of the N sample displacements to a random real

number between -½ and +½.

• The resulting vibrations. After the string is plucked, the string vibrates. The pluck causes a displacement which

spreads wave-like over time. The Karplus-Strong algorithm simulates this vibration by maintaining a ring buffer of

the N samples: the algorithm repeatedly deletes the first sample from the buffer and adds to the end of the buffer the

average of the first two samples, scaled by an energy decay factor of 0.996.

Why it works? The two primary components that make the Karplus-Strong algorithm work are the ring buffer feedback

mechanism and the averaging operation.

• The ring buffer feedback mechanism. The ring buffer models the medium (a string tied down at both ends) in which

the energy travels back and forth. The length of the ring buffer determines the fundamental frequency of the resulting

sound. Sonically, the feedback mechanism reinforces only the fundamental frequency and its harmonics (frequencies

at integer multiples of the fundamental). The energy decay factor (.996 in this case) models the slight dissipation in

energy as the wave makes a roundtrip through the string.

• The averaging operation. The averaging operation serves as a gentle low pass filter (which removes higher

frequencies while allowing lower frequencies to pass, hence the name). Because it is in the path of the feedback, this

has the effect of gradually attenuating the higher harmonics while keeping the lower ones, which corresponds closely

with how an actual plucked string sounds.

Functional Specifications:

DblQueue class: This class will provide support for an unbounded queue abstraction which holds double values. The

class should use a linked list as its underlying representation. It must support the following operations:

DblQueue() The default constructor that creates an empty queue.

DblQueue(const DblQueue& rhs) The copy constructor.

~DblQueue() The destructor.

const DblQueue& operator= (const DblQueue& rhs) The assignment operator.

bool isEmpty( ) const Tests whether the queue is empty.

void enqueue(const ItemType& item) Add an item to the end of the queue.

void dequeue( ) Remove the item at the front of the queue. If a dequeue( ) is attempted on an

empty queue then throw a std::underflow_error exception.

ItemType front( ) const Return the item at the front of the queue without removing it. If a front( ) is

attempted on an empty queue then throw a std::underflow_error exception.

size_t size( ) const Return the number of items on the queue.

Note 1: Be sure to name your files, classes, and methods exactly as specified; in particular, the files are named

DblQueue.h & DblQueue.cpp, with capital 'D' and 'Q', and the class is named DblQueue.

Note 2: The DblQueue.h file should use a typedef to define the new type ItemType. You should use this data type

whenever you specify the type of data that your queue will be holding. If you ever want to change the type of data that the

queue holds, you will only need to change one line of code.

Note 3: Your implementation is to be based on a singly linked list.

Note 4: The five methods of the stack class (enqueue, dequeue, front, isEmpty, & size) should all operate in O(1) time.

Note 5: To throw an underflow exception, you would write code such as:

throw std::underflow_error("The queue is empty");

Note that there is no “new” operation as needed when throwing exceptions in Java.

Other details: Here are a few notes that might be helpful:

1. Continue to use type size_t when appropriate.

2. Use the base initialization list for your constructors whenever possible.

3. You are free to add private helper methods to the class. You are not allowed to change the public interface of the

class.

4. It is true that for representing a guitar string it would be sufficient to use a bounded queue, since the frequency of a

guitar string does not change once it has been created. However, you are required to implement an unbounded queue,

as you will need it for the next assignment.

GuitarString class: This class will model/simulate a vibrating guitar string of a given frequency. Implement a class

named GuitarString with the following API:

GuitarString(double frequency) The constructor for a string of the given frequency. Throw a std::invalid_argument

exception if the frequency is not positive.

void pluck() Fill the string’s buffer with white noise.

void tic() Advance the simulation one time step.

double sample() const Return the current sample.

size_t getTime() const Return number of tics.

double getFrequency() const Return the frequency of the string.

Each GuitarString object will need an internal queue (use a DblQueue) to act as its ring buffer feedback mechanism.

Constructor: create a ring buffer of the desired capacity N (sampling rate divided by frequency, rounded to the nearest

integer) by initializing a queue and filling the queue to represent a guitar string at rest by enqueueing N zeros.

pluck: Replace the N elements in the ring buffer (queue) with N random values between -0.5 and +0.5 (code to generate

random values is given below).

tic: Apply the Karplus-Strong update: delete the sample at the front of the buffer and add to the end of the buffer the

average of the deleted sample & the next sample, multiplied by the energy decay factor. A tic represents that our string

has been told to advance one time increment in its simulation.

sample: Return the value of the item at the front of the buffer.

getTime: Return the total number of times tic() was called since the object was created. This provides information on

how long the simulation of the current string has been running.

getFrequency: Return the frequency that was specified in the constructor.

Note 1: Be sure to name your files, classes, and methods exactly as specified; in particular, the files are named

GuitarString.h & GuitarString.cpp, with capital 'G' and 'S', and the class is named GuitarString.

Note 2: There is no default constructor for this class.

Note 3: If your GuitarString class performs any dynamic memory allocation, be sure to include the Big 3.

Note 4: You can add private data members to the class as you see fit. You are free to add private helper methods to the

class, but you are not allowed to change the public interface of the class.

Note 5: To generate a random number in the range -0.5 to +0.5 you can use the random number generation facilities

provided by C++ (this is new in C++11). These facilities are defined in <random>, so you will need to perform the

appropriate #include. You will need to create a generator object. This object will be capable of generating pseudo-

random numbers for you (they are pseudo-random since you get the same sequence of random numbers when started with

the same seed). Thus it is standard practice to seed the random number generator with a value based on time (which

requires #include <chrono>). After creating the generator object, you will need to create a distribution object. This

object directs what kind of random numbers will be generated. In our case we want double values in the range -0.5 to

+0.5. Once we have these two objects, we can get the desired random numbers by invoking the distribution object and

passing it the generator object. Here is the code you can use:

#include <chrono>

#include <random>

...

// then later in your code where you want to generate random numbers, do:

// create the seed

unsigned seed = (unsigned) std::chrono::system_clock::now().time_since_epoch().count();

// create the generator using the seed

std::default_random_engine generator(seed);

// create the appropriate distribution

std::uniform_real_distribution<double> distribution(-0.5, 0.5);

// create random numbers as needed (the following will likely be in a loop)

double num = distribution(generator);

You should create the seed, generator, and distribution object only once per pluck or once per program execution. You

can then use the distribution object as many times as needed to create random numbers. You can see an example of using

these functions on this page: http://www.cplusplus.com/reference/random/uniform_real_distribution/.

Main: The main method in GuitarHero.cpp currently performs the following actions: (1) It creates an array of 3

octaves worth of GuitarString pointers, and dynamically creates a GuitarString object for each array element. These guitar

strings represent a total of 37 notes on the chromatic scale from 110Hz to 880Hz. See the diagram below for a mapping of

array index to note. (2) The program then prompts the user for the name of the input & output files. It opens the files and

primes the output file with the first two required comment lines of a .dat file used by sox. (3) The program then closes all

files and frees all the created guitar strings. Between steps 2 & 3 is where you will need to do your work of reading the

input file and generating data for the output file.

The input file is an ASCII text file. Each line of the file contains two numeric values. The first value of each line is a

double value indicating a time (in seconds) when the corresponding guitar string is to be plucked. The lines of the file

must be in chronological order of this time value. The second value of each line is an integer value in the range -1 to 36.

The values 0-36 are indices into the array of guitar strings, indicating which string is to be plucked. The value -1 is a

special “end song” value which indicates that the song should end at the specified time (this allows you to specify a fadeout

period for your last note played). Here is sample data file that plays a concert C scale (C-E-G-C). The first C note

(index 3) is played at a half second, the E note is played at 1 second, the G is played at 1.5 seconds, and the final C note is

played at 2 seconds. These notes die out over the next second until we hit the “end song” flag at the 3 second mark.

Your task is to read the input file and generate the desired .dat output file. Recall from the last assignment that digitized

sound is sampled at 44100 samples per second. That implies that your output file should contain 44100 lines of output for

each second of sound generated. You will need to write some type of loop nest that simulates the passing of time (e.g.,

increments a clock counter (type double) in steps of size 1/44100), and also reads all the data in the input file (you process

the line of data when the clock counter reaches the specified time). If the time for the next note has not been reached, you

need to sample & tic all guitar strings, and produce a single line of output in the output file. As you sample all the strings,

you need to sum up all the samples to produce a single cumulative sample which is the sound value you write to the

output file for that time step. When the time for the next note has been reached, you pluck the specified string (unless you

reach the “end song” value in which case you stop generating any more sounds). You continue this process until the time

for the “end song” value is encountered or the input data file is exhausted.

Once you have a .dat file generated, use the sox program to create a corresponding .wav file (just like the last assignment)

and listen to it with your computer’s media player. You can ignore messages from sox about clipped samples. As we

generate sounds, those sounds represent the culmination of all the strings, and thus some sound values may be outside the

range -1.0 to +1.0. When sox encounters such values it must cut them down into the range -1.0 to +1.0 – this is called

“clipping”.

Sox can be downloaded from: http://prdownloads.sourceforge.net/sox/sox12181.zip?download

Other details:

Here are a few notes that you need to keep in mind:

1. The array of strings contains GuitarString pointers. You need to dereference the pointers to access the GuitarString

objects. This is accomplished most easily with arrow notation; e.g., strings[i]->tic();

2. You should verify all data read from the input file. The time values need to be in chronological order and the note

values must be in the range -1 to 36 (inclusive). If an input value is not valid, generate a std::out_of_range

exception. Note that it is valid for adjacent lines to have the same time value, indicating that those strings are plucked

simultaneously.

3. The .dat output file has the same format as the previous assignment. Please review the spec and your code for that

assignment. The data you generate in the .dat file should start at time zero (which may not be the first time specified

in the input file), increase by the time step value (1/44100), and finally stop at the time of the last note (or the special

“end song” indicator).

4. You may assume the input file always has pairs of values. That is, if you are able to extract a time value then there

will always be an associated note value. You may also assume the input file has at least one pair of values.

5. You should write helper functions to isolate complex logic and to make your program easier to read/understand. Each

function should be a reasonable length (less than 40 lines of code).

6. The DblQueue class is needed for the GuitarString class. You do not need to use the DblQueue class for any other

processing in this assignment; in particular you should not use queues as temporary storage as you read data from the

input file or as you generate data for the output file – rather you should read data only as it is needed and you should

write data to the output file as it is generated.

7. You may want your program to give some indication that it is making progress since generating a large .dat file will

be time consuming. Do this in a way so as to not overwhelm the user. I.e., generate a dot or two for each pluck of a

string or for each full/partial second of music generated, but not for each time step that is computed (since there are

44.1k steps per second of music generated).

Logistics:

• Electronic Turn-in: You should turn in the following files, named as follows:

o GuitarHero.cpp

o DblQueue.h, DblQueue.cpp, and DblQueueTest.cpp

o GuitarString.h, GuitarString.cpp and GuitarStringTest.cpp

o a README document (a .txt text file or a .doc Word document), containing the answers to each writeup

question below.

• Each file you turn in should have the standard block comments at the top of the file, including an honor statement.

Be sure to use good programming style.

• You must implement the linked-list queue by writing your own queue code - you may not use any classes from

the C++ standard template library to do the work.

Electronic turn-in should be done via the assignment page in Brightspace. Be sure to attach all the required files prior to

hitting the "Submit" button. Please do not zip up your entire project directory for submission, as the file would be

extremely large – just submit the six required files.

Write-up Questions:

Answer the following questions in the README text file you are to submit for grading.

1. State your name and email address.

2. After reviewing this spec and the provided files, please estimate/report how long you think it will take you to

complete it.

3. How many hours did you actually spend total on this assignment?

4. Who/What did you find helpful for this project?

5. How did you test that the final program is correct?

6. How did you test that your queue implementation is correct?

7. What did you enjoy about this assignment? What did you hate?

8. Do you have any suggestions for improving this assignment?

Please contact me if you find errors in this document or you feel something is missing or unnecessarily confusing.

Feel free to share any input files you create that generate interesting music by posting them to Piazza.

Sample Execution:

Enter the name of the input file: song.txt

Enter the name of the output file: song.dat

Reading the input file and generating a .dat file for sox

..............................Done.

<Note: the number of “progress” dots produced will depend on how many you print per progress milestone, and how you define such

milestone.>

Sharing song files:

Past students have had a great deal of fun creating songs for their GuitarHero program. I even gave extra credit to the two

songs that received the most votes in a poll. Unfortunately, with the increase enrollment of the class, it is difficult for me

to manage all song submissions and so there is no longer an extra credit option. But you are free to share your songs with

the rest of the class. Please post your songs to Piazza (post it as a Note rather than a Question). You can post as many song

files as you wish. Please limit songs to 30 seconds or less in playing time. Please only post the .txt files -- do not post the

.dat or the .wav files. You can post anonymously if you want.

Let your inner Guitar Hero go wild!!

Past student feedback from README file:

“I do agree with Professor Roth emphasizing for us to read the spec file many times, because it is essential to implement

GuitarHero.cpp”

Additional information: More information on musical sounds and instruments can be found here.

Acknowledgments:

This assignment is a modification of a submission made by Kevin Wayne to the Nifty Assignments collection. It was originally designed at Princeton

University by Andrew Appel, Jeff Bernstein, Maia Ginsburg, Ken Steiglitz, Ge Wang, and Kevin Wayne.




站长地图