Thursday, 15 September 2016

Chapter 6 // Drill 2, 3, 4, 5 - Principles & Practice Using C++

In all these exercises I am using Visual Studio Community 2015 and the header file "std_lib_facilities.h" which can be found here:


http://www.stroustrup.com/Programming/PPP2code/std_lib_facilities.h


My version is spelt differently so adjust the code accordingly if copying and pasting.


Chapter 6 // Drill 2

Change the character used as the exit command from q to x.

#include "stdafx.h"
#include "std_lib_facilities_new_version.h"
using namespace std;

double val; // messy way for main to access val in get()

//------------------------------------------------------------------------------

class Token{
public:
char kind;        // what kind of token
double value;     // for numbers: a value 
Token(char ch)    // make a Token from a char
:kind(ch), value(0) { }
Token(char ch, double val)     // make a Token from a char and a double
:kind(ch), value(val) { }
};

//------------------------------------------------------------------------------

class Token_stream {
public:
Token_stream();   // make a Token_stream that reads from cin
Token get();      // get a Token (get() is defined elsewhere)
void putback(Token t);    // put a Token back
private:
bool full;        // is there a Token in the buffer?
Token buffer;     // here is where we keep a Token put back using putback()
};

//------------------------------------------------------------------------------

// The constructor just sets full to indicate that the buffer is empty:
Token_stream::Token_stream()
:full(false), buffer(0)    // no Token in buffer
{
}

//------------------------------------------------------------------------------

// The putback() member function puts its argument back into the Token_stream's buffer:
void Token_stream::putback(Token t)
{
if (full) error("putback() into a full buffer");
buffer = t;       // copy t to buffer
full = true;      // buffer is now full
}

//------------------------------------------------------------------------------


Token Token_stream::get()
{
if (full) 
{       // do we already have a Token ready?
 // remove token from buffer
Token_stream::full = false;
return buffer;
}


char ch;
cin >> ch;    // note that >> skips whitespace (space, newline, tab, etc.)

switch (ch) 
{
case ';':    // for "print"
case 'x':    // for "quit"

case '(': case ')': case '+': case '-': case '*': case '/':
return Token(ch);        // let each character represent itself
case '.':
case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9':

{
cin.putback(ch);         // put digit back into the input stream
cin >> val;              // read a floating-point number
return Token('#', val);   // let '#' represent "a number"
}

default:
error("Bad token");
}
}

//------------------------------------------------------------------------------


Token_stream ts;        // provides get() and putback() 

//------------------------------------------------------------------------------

double expression();    // declaration so that primary() can call expression()

//------------------------------------------------------------------------------

// deal with numbers and parentheses
double primary()
{
Token t = ts.get();

switch (t.kind) 
{
case '(':    // handle '(' expression ')'
{
double d = expression();
t = ts.get();
if (t.kind != ')') error("')' expected)");
return d;
break;
}

case '#':            // we use '8' to represent a number
{
return t.value;  // return the number's value
break;
}

default:
error("primary expected");
}
}

//------------------------------------------------------------------------------

// deal with *, /
double term()
{
double left = primary();
Token t = ts.get();        // get the next token from token stream

while (true) 
{
switch (t.kind) 
{
case '*':
{
left *= primary();
t = ts.get();
break;
}

case '/':
{
double d = primary();
if (d == 0) error("divide by zero");
left /= d;
t = ts.get();
break;
}

default:
ts.putback(t);     // put t back into the token stream

return left;
}
}
}

//------------------------------------------------------------------------------

// deal with + and -
double expression()
{
double left = term();      // read and evaluate a Term
Token t = ts.get();        // get the next token from token stream

while (true) {
switch (t.kind) {
case '+':
left += term();    // evaluate Term and add
t = ts.get();
break;
case '-':
left -= term();    // evaluate Term and subtract
t = ts.get();
break;
default:
ts.putback(t);     // put t back into the token stream
return left;       // finally: no more + or -: return the answer
}
}
}

//------------------------------------------------------------------------------

int main()
try
{
val = 0;
while (cin)
{
Token t = ts.get();

while (t.kind == '=')
t = ts.get();

if (t.kind == 'x')
{
return 0;
}

ts.putback(t);
cout << expression() << '\n';
}
return 0;
}

catch (exception& e) {
cerr << "error: " << e.what() << '\n';
keep_window_open();
return 1;
}

catch (...) {
cerr << "Oops: unknown exception!\n";
keep_window_open();
return 2;
}


//------------------------------------------------------------------------------


It took a few things to get this working properly. It wasn't just as simple as changing line 87:
case 'q':    // for "quit"
to an x instead. 

First I changed the case to handle 'x' but I noticed that when you pressed x and enter, an error would pop up and force you to exit instead of just quitting the program. Clearly something wasn't right. Pressing 'x' followed by ';' worked and quit the while loop in main but we're supposed to press ';' to print something.

I noticed that in main, to use 'x' to quit it calls ts.get() and assigns 'x' to t.kind. Now main evaluates if what you have inputted is a primary, primary() only handles '(' and '#' so primary() doesn't know what to do with it. I just added another case that handles 'x', returns 'x' so that main just quits the loop when it's returned instead of defaulting to the error message found in primary(). 


That said, doing it that way means you do have to enter x twice and I spent almost 2 hours trying to figure out why but I still don't know. Through debugging I noticed that after an initial calculation has been performed you HAVE to provide at least 2 char's for main to work through. get() always goes through 2 char's (apart from the first time you open the program and 0 is already in the input buffer). I've tried putting numbers into the input buffer if an 'x' is entered but it still makes you put an x in again to quit. 

Looking ahead to chapter 7 Bjarne sort of addresses it and using the code he's written I've managed to get this working now (it's all to do with main) but I don't fully understand why. I guess it's just one of those things I won't get until later on.

Chapter 6 // Drill 3

Change the character used as the print command from ';' to '='.

This one is pretty simple, just changing line 68 to case '=':, then line 203 to compare to =. I also removed the "=" in the cout underneath so it made a bit more sense when printing out.


Chapter 6 // Drill 4

Add a greeting line in main():
     "Welcome to our simple calculator.
     Please enter expressions using floating-point numbers."

int main()
try
{
cout << "Welcome to our simple calculator!\n";
cout << "Please enter expressions using floating-point numbers:\n";

val = 0;
while (cin)
{
Token t = ts.get();

while (t.kind == '=')
t = ts.get();

if (t.kind == 'x')
{
return 0;
}

ts.putback(t);
cout << expression() << '\n';
}
return 0;

}


Chapter 6 // Drill 5

Improve that greeting by mentioning which operators are available and how to print and exit.

//------------------------------------------------------------------------------

void printInstructions()
{
cout << "Welcome to our simple calculator!\n";
cout << "You can use + - / *\n";
cout << "Press = to print the answer and x to exit.\n";
cout << "Please enter expressions using floating-point numbers:\n" << endl;
}

//------------------------------------------------------------------------------

int main()
try
{
printInstructions();

val = 0;
while (cin)
{
Token t = ts.get();

while (t.kind == '=')
t = ts.get();

if (t.kind == 'x')
{
return 0;
}

ts.putback(t);
cout << expression() << '\n';
}
return 0;

}


4 comments:

  1. Please help me!

    "So, we are ready to test our second version. This second version of the calculator program (including Token_stream) is available as file calculator01.cpp. Get it to run and try it out."

    I just can't find "calculator01.cpp" anywhere on his website: http://www.stroustrup.com/Programming/

    Where did you find it?

    ReplyDelete
    Replies
    1. For the creator of c++ I find it rather amusing at just how unorganised his website is. I simply just put calculator01.cpp into google. I can't find calculator01, this is the only version I could find.

      http://stroustrup.com/Programming/calculator02buggy.cpp

      Delete
  2. I am completely stuck because of this... I wish I could hit Stroustrup with a baseball bat. I want him dead.

    ReplyDelete
  3. I finally got calculator01 here: http://people.ds.cam.ac.uk/nmm1/C++/Exercises/Chapter_06/calculator02.cpp

    Note: A bug-free version of calculator02buggy is actually calculator01

    ReplyDelete