関数atoi
(strtol
または同様の関数)を使用する場合、整数変換が失敗したかどうか、または変換されていたC文字列がであったかどうかをどのように判断できます0
か?
私がやっていること0
は、許容可能な値であり、変換されるC文字列には任意の数の0
sが含まれる可能性があります。また、先頭に空白がある場合もあります。
適切な関数 (C スタイルの関数を使用することを主張している限り) は次のとおりでstrtol
あり、変換コードは次のようになります。
const char *number = "10"; /* for example */
char *end;
long value = strtol(number, &end, 10);
if (end == number || *end != '\0' || errno == ERANGE)
/* ERROR, abort */;
/* Success */
/* Add whatever range checks you want to have on the value of `value` */
いくつかのコメント:
strtol
実際の数値の前の空白を許可します (意味: 静かにスキップします)。このような先頭の空白をエラーとして扱う場合は、自分で確認する必要があります。
のチェックは*end != '\0'
、数字の後に何もないことを確認します。実際の数字の後に他の文字 (空白?) を許可したい場合は、それに応じてこのチェックを変更する必要があります。
PSend == number
後でチェックを追加して、空の入力シーケンスをキャッチしました。「すべての空白」および「数字がまったくない」入力は、*end != '\0'
チェックだけでキャッチされます。ただし、事前に空の入力をキャッチすることは理にかなっています。その場合end == number
はチェック不要となる場合があります。
これはc++とタグ付けされているため:
template< typename T >
inline T convert(const std::string& str)
{
std::istringstream iss(str);
T obj;
iss >> std::ws >> obj >> std::ws;
if(!iss.eof())
throw "dammit!";
return obj;
}
文字列から整数への変換のための go-to 関数は になりましstoi
た。これは を受け取ってstring
を返すint
か、エラー時に例外をスローします。
istringstream
受け入れられた回答で言及されている冗長なハックはもう必要ありません。
(それぞれ/ / / /変換用のstol
/ stoll
/ stof
/ stod
/もあります。)stold
long
long long
float
double
long double
strtol()のマニュアルページから:
endptrがNULLでない場合、strtol()は最初の無効な文字のアドレスを*endptrに格納します。ただし、数字がまったくない場合、strtol()はnptrの元の値を*endptrに格納します。(したがって、* nptrがそうではなく
'\0'
、** endptrが'\0'
戻ってきた場合、文字列全体が有効でした。)
C/C++ を使ってからしばらく経ちましたが、(非常に) 単純な解決策は、文字列の "0" だけをチェックすることだと思われます。
int value = atoi(string_number.c_str());
if ( !value && string_number != "0" ) {
// error
} else {
// great success!
}
少し重いstrtol
ですが、の代替手段は次のとおりです。sscanf
const char *numStr = "12345"; // input string
int value;
if(sscanf(numStr, "%d", &value) == 1)
; // parsing succeeded, use value
else
; // error
strtol()
ただし、これにより、文字列の先頭に空白を含めることができ (これは望ましい場合とそうでない場合があります)、数字の後ろに何かを置くことができるため、"123abc" が受け入れられて 123 が返されます。AndreyT が示すように。