L'estrazione normale di un valore intero ha esito positivo se il flusso è in grado di leggere qualsiasi valore intero. Cioè, se c'è almeno una cifra seguita facoltativamente da qualcosa, la lettura di un intero ha esito positivo. Le normali operazioni di estrazione non provano a leggere, in particolare non cercano di trovare il successivo spazio bianco.
Dal suono di ciò, si vuole essere sicuri che ci sia uno spazio vuoto che segue il proprio numero e fallisce se non c'è. Posso pensare a due diversi approcci per fare questo:
- Creare un semplice manipolatore che controlli la presenza del flusso su un carattere di spazio bianco. Ciò, tuttavia, significa che si leggeranno i valori usando qualcosa come
in >> value >> is_space
.
- Creare un facet personalizzato
std::num_get<char>
, installarlo in un std::locale
e imbue()
questo std::locale
nel tuo stream (s). È un po 'più coinvolto, ma non richiede modifiche al modo in cui vengono letti gli interi.
Creazione di un manipolatore come questo è abbastanza banale:
std::istream& is_space(std::istream& in)
{
if (!std::isspace(in.peek()))
{
in.setstate(std::ios_base::failbit);
}
return in;
}
Ora, cambiando i numeri modo in cui vengono letti è più interessante e ho il sospetto che avevo appena nominato un certo numero di classi della libreria standard la maggior parte delle persone sono abbastanza inconsapevole di. Quindi, digitiamo rapidamente un esempio anche per questo. Cambierò la faccetta std::num_get<char>
solo per gestire unsigned int
: per farlo per altri tipi interi è necessario sovrascrivere più funzioni.Quindi, ecco un sostituto per il std::num_get<char>
sfaccettatura:
class num_get:
public std::num_get<char>
{
iter_type do_get(iter_type it, iter_type end,
std::ios_base& ios, std::ios_base::iostate& err,
unsigned int& value) const
{
it = std::num_get<char>::do_get(it, end, ios, err, value);
if (it != end && !isspace(static_cast<unsigned char>(*it)))
{
err |= std::ios_base::failbit;
}
return it;
}
};
Tutto ciò che fa è per derivare una classe da std::num_get<char>
e ignorare una delle sue funzioni virtuali. L'implementazione di questa funzione è abbastanza semplice: inizia con la lettura del valore delegando alla classe base (mi sono appena reso conto che le funzioni virtuali in realtà vogliono essere protette piuttosto che private come ho fatto in passato ma questa è una discussione completamente diversa) . Indipendentemente dal fatto che questo abbia avuto successo (se non lo fosse, avrà impostato erroneamente uno stato di errore) i controlli di override se è disponibile un altro carattere e, in tal caso, controlla se si tratta di uno spazio e se non imposta uno std::ios_base::failbit
in il risultato dell'errore err
.
ciò che rimane è per impostare il flusso di utilizzare questo particolare aspetto in un std::locale
e agganciare il nuovo std::locale
in un flusso:
std::locale loc(std::locale(), new num_get);
in.imbue(loc);
Le std::locale
s ed i suoi aspetti sono internamente riferimento contati, cioè si shouldn 't tenere traccia del puntatore al facet e non è necessario mantenere il std::locale
in entrambi. Se sembra essere complicato a il std::locale
creato o si desidera utilizzare questa logica modificata ovunque, è possibile impostare il globale std::locale
che viene utilizzato per inizializzare qualsiasi flusso appena creato per utilizzare il facet personalizzato std::num_get<char>
.
È proprio come funziona 'operator >>'; estrarrà '33' e quindi si fermerà, 'abcd' rimarrà nello stream per la prossima chiamata a 'operator >>'. Si potrebbe invece leggere '33abcd4' in una stringa e quindi verificare la presenza di caratteri non numerici. Inoltre, se hai un compilatore recente che supporta C++ 11, controlla se la libreria standard fornisce 'std :: stoull'. – jrok