斯特劳斯特鲁普的C++书练习

Stroustrup's C++ Book exercise

本文关键字:C++ 练习 斯特劳斯      更新时间:2023-10-16

我正在从Stroustrup的书中学习c++: 编程:使用c++的原理和实践。我正在读第6章我们正在写一个计算器,现在:

我必须添加一个比*/更强的阶乘子句。在最初的程序中,我们有3个级别,primary(浮点文字&括号),terms(*和/)和expressions(+或-)。expression所做的是调用term,收集它的左侧,以此类推。为了实现阶乘,我在termprimary()之间添加了一个fact(),以使其更紧凑。在添加fact()之前,计算器工作得很好。然后我添加了fact,以便从primaryterm获取fact的数据。

现在进入问题:

Q1:正如我已经说过的,在添加fact之前一切都很好。现在输出的唯一正确的方程是term。当我尝试做其他算术时,它只打印最后输入的数字。我把其他的操作弄乱了。

Q2:(有点离题)为什么当我试图退出程序时(这是通过输入'q',它需要我三到四次进入'q',直到它退出。

代码如下:

#include "std_lib_facilities.h"

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
    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();
};
// Constructor
Token_stream::Token_stream()
    :full(false),buffer(0)
{
}
Token_stream ts;
void Token_stream::putback(Token t)
{
    if (full)  error("putback() into a full buffer"); // checks if we're using putback() twice
    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
        full = false;
        return buffer;
    }
    char ch;
    cin >> ch; // note that >> skips whitespace
    switch(ch) {
    case ';':   // for "print"
    case 'q':   // for "quit"
    case '(': case ')': case '{': 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
        double val;
        cin >> val;
        return Token('8',val);  // let '8' represent a "number"
        break;
    }
    default:
        error("Bad token");
        return 0;
    }
}

Token get_token()    // read a token from cin
{
    char ch;
    cin >> ch;    // note that >> skips whitespace (space, newline, tab, etc.)
    switch (ch) {
    case 'q':
    case ';':
    case '(': case ')': case '{': 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
            double val;
            cin >> val;              // read a floating-point number
            return Token('8',val);   // let '8' represent "a number"
        }
    default:
        error("Bad token");
        return 0;
    }
}
//------------------------------------------------------------------------------
double expression();  // read and evaluate a Expression
//------------------------------------------------------------------------------
double term();        // read and evaluate a Term
//------------------------------------------------------------------------------
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;
        }
    case '{':    // handle '(' expression ')'
            {
                double d = expression();
                t = ts.get();
                if (t.kind != '}') error("'}' expected");
                return d;
            }
    case '8':            // we use '8' to represent a number
        return t.value;  // return the number's value
    default:
        return 0;
        error("primary expected");
    }
}
//------------------------------------------------------------------------------
int main()
try {
        double val = 0;
        while(cin)
        {
            Token t = ts.get();
            if(t.kind == 'q') break; //'q' for quit
            if(t.kind == ';')
                cout << "=" << val << "n";
            else
                ts.putback(t);
            val = expression();
        }
    keep_window_open("q");
}
catch (exception& e) {
    cerr << e.what() << endl;
    keep_window_open ("q");
    return 1;
}
catch (...) {
    cerr << "exception n";
    keep_window_open ("q");
    return 2;
}
//------------------------------------------------------------------------------
double expression()
{
    double left = term();      // read and evaluate a Term
    Token t = ts.get();        // get the next token
    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);
            return left;       // finally: no more + or -: return the answer
        }
    }
}
//------------------------------------------------------------------------------
double factorial(double val)
{
    double res=1;
    for(int i=1; i<=val; i++)
        res *= i;
    return res;
}
double fact()
{
    double left = primary();
    Token t = ts.get();
    switch(t.kind)
    {
    case '!':
        {   double res = factorial(left);
            return res;
        }
    default:
        return left;
    }
}
//------------------------------------------------------------------------------
double term()
{
    double left = fact();
    Token t = ts.get();     // get the next token
    while(true) {
        switch (t.kind) {
        case '*':
            left *= fact();
            t = ts.get();
            break;
        case '/':
            {
                double d = fact();
                if (d == 0) error("divide by zero");
                left /= d;
                t = ts.get();
                break;
            }
        default:
            ts.putback(t);
            return left;
        }
    }
}

对不起,如果我不够具体。这是我第一次发这样的帖子。

从你的第二个问题开始:

您需要按q几次,因为您在

中将其作为参数:
keep_window_open("q");

所以,每次你退出你的while循环,它等待你进入q以便关闭控制台。

关于你的阶乘函数:

指定你的计算器语法:写出现有操作的优先级顺序,即包含它们的函数的调用顺序。这将使您更容易合并新功能。

factorial作为额外的case放入primary()将是一个好主意,因为括号乘法,除法等term()中具有更高的优先级(称为早期)。

这是一个可能的实现:

double primary(){
    Token t = ts.get();
    switch (t.kind) {
    case '{':{
        double d = expression();
        t = ts.get();
        if (t.kind != '}') error("'}' expected");
        return d;
    }
    case '(':    // handles '(' expression ')'{    
        double d = expression();
        t = ts.get();
        if (t.kind != ')') error("')' expected");
        return d;
    }
    case '8': case '!':{
        // include a test whether the number is integer and > 0 
        if(is_factorial()){
            double d  = factorial(t.value);
            t = ts.get();
            return d;
        }
        else return t.value;
    }
    default:
        error("primary expected");
    }
}

地点:

/*
   Non-member method: factorial.
   Use: double fact = factorial(double);
   This funnction provides factorial operator.
*/
double factorial(double num){
   if(num <= 1) return 1;
   return num*factorial(num-1);
}

/*
   Non-member method: is_factorial.
   Use: bool fact = is_factorial(void);
   This funnction returns true if a number
   is followed by factorial opertor. 
   Used as an indicator to call factorial function.
*/
bool is_factorial(){
    Token t = ts.get();
    if (t.kind == '!'){
        ts.putback(t);
        return true;
    }
    ts.putback(t);
    return false;
}