std :: stringstream strano comportamento
Domanda
Alcune informazioni di base, per un compito a casa ho dovuto scrivere una calcolatrice di notazione polacca usando alberi binari, per farlo ho dovuto analizzare l'input della riga di comando in modo da costruire correttamente l'albero binario e poi passarci sopra per dare una risposta valida all'espressione matematica che è stata inserita.
Per l'analisi ho usato uno std :: stringstream in modo da poter facilmente convertire lo std :: string mi è stato consegnato un float valido (o intero, doppio). Il problema che ho riscontrato è stato il seguente codice, che mostra l'errore e come ho risolto il problema. Speravo che qualcuno dove sarebbe stato in grado di dirmi se stavo facendo qualcosa di sbagliato e .clear () non è corretto, o se questo è un bug nella libreria standard nel modo in cui gestisce questo particolare input (succede solo per + e -).
#include <iostream>
#include <sstream>
#include <string>
int main() {
std::string mystring("+");
int num;
char op;
std::stringstream iss(mystring);
iss >> num;
// Seems it is not a number
if (iss.fail()) {
// This part does not work as you would expect it to
// We clear the error state of the stringstream
iss.clear();
std::cout << "iss fail bit: " << iss.fail() << std::endl;
iss.get(op);
std::cout << "op is: " << op << " iss is: " << iss.str() << std::endl;
std::cout << "iss fail bit: " << iss.fail() << std::endl;
// This however works as you would expect it to
std::stringstream oss(iss.str());
std::cout << "oss fail bit: " << oss.fail() << std::endl;
oss.get(op);
std::cout << "op is: " << op << " oss is: " << oss.str() << std::endl;
std::cout << "oss fail bit: " << oss.fail() << std::endl;
} else {
// We got a number
}
}
Esempio di output dal programma:
iss fail bit: 0
op is: iss is: +
iss fail bit: 1
oss fail bit: 0
op is: + oss is: +
oss fail bit: 0
Forse vedrete qualcosa che mi è sfuggito, o se questo è davvero un bug più in alto rispetto al mio programma, nel qual caso i consigli su dove segnalare questo sarebbe molto apprezzato.
Soluzione
Quando dici:
iss.clear();
std::cout << "iss fail bit: " << iss.fail() << std::endl;
iss.get(op);
stai provando a leggere qualcosa che è già stato letto. È necessario ripristinare il puntatore di lettura dei flussi:
iss.clear();
iss.seekg(0); // start again
std::cout << "iss fail bit: " << iss.fail() << std::endl;
iss.get(op);