回答:
以下は、文字列を数値として解析する際の限られた精度に依存しない方法の1つです。
NSCharacterSet* notDigits = [[NSCharacterSet decimalDigitCharacterSet] invertedSet];
if ([newString rangeOfCharacterFromSet:notDigits].location == NSNotFound)
{
// newString consists only of the digits 0 through 9
}
+[NSCharacterSet decimalDigitCharacterSet]
およびを参照してください-[NSString rangeOfCharacterFromSet:]
。
[[NSCharacterSet characterSetWithCharactersInString:@"0123456789."] invertedSet]]
invertedSet
作成されたセットをフォームでラップしたクラスを返し、すべてのクエリに対して反対の値を返すのではないかと思います。しかし、私にははっきりとはわかりません。
NSNumberFormatterクラスのnumberFromString:
メソッドを使用することをお勧めします。数値が無効な場合、nilが返されます。それ以外の場合は、NSNumberを返します。
NSNumberFormatter *nf = [[[NSNumberFormatter alloc] init] autorelease];
BOOL isDecimal = [nf numberFromString:newString] != nil;
NO
文字列を返します。また、知っておく価値があります:YES
空白で埋められた数値も返されます。ところで、私はあなたの答えに実際のコードスニペットを追加しました、気にしないでください。
正規表現、パターン"^[0-9]+$"
、次の方法で検証します-validateString:withPattern:
。
[self validateString:"12345" withPattern:"^[0-9]+$"];
"^[0-9]+(.{1}[0-9]+)?$"
"."
。
"^[0-9]{4}$"
。"."
、長さは2〜5の間です。
"^[0-9]{2,5}$"
。"^-?\d+$"
正規表現は、オンラインWebサイトで確認できます。
ヘルパー関数は以下の通りです。
// Validate the input string with the given pattern and
// return the result as a boolean
- (BOOL)validateString:(NSString *)string withPattern:(NSString *)pattern
{
NSError *error = nil;
NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:pattern options:NSRegularExpressionCaseInsensitive error:&error];
NSAssert(regex, @"Unable to create regular expression");
NSRange textRange = NSMakeRange(0, string.length);
NSRange matchRange = [regex rangeOfFirstMatchInString:string options:NSMatchingReportProgress range:textRange];
BOOL didValidate = NO;
// Did we find a matching range
if (matchRange.location != NSNotFound)
didValidate = YES;
return didValidate;
}
遊び場でテストします。
import UIKit
import Foundation
func validate(_ str: String, pattern: String) -> Bool {
if let range = str.range(of: pattern, options: .regularExpression) {
let result = str.substring(with: range)
print(result)
return true
}
return false
}
let a = validate("123", pattern: "^-?[0-9]+")
print(a)
func numberOnly(string: String) -> Int { let expression = "" let regex = NSRegularExpression.init(pattern: expression, options: .caseInsensitive) let numberOfMatches = regex.numberOfMatches(in: string, options: .reportProgress, range: NSRange.init(location: 0, length: string.characters.count)) if numberOfMatches == 0 { return Int(string)! } return 0 }
エラーが発生しましたPlayground execution aborted: error: Execution was interrupted, reason: EXC_BAD_INSTRUCTION (code=EXC_I386_INVOP, subcode=0x0).
return matchRange.location != NSNotFound;
^-?\d+$
、私はサイトでそれを確認しました:regex101.com
NSScannerを作成して、文字列をスキャンするだけです。
NSDecimal decimalValue;
NSScanner *sc = [NSScanner scannerWithString:newString];
[sc scanDecimal:&decimalValue];
BOOL isDecimal = [sc isAtEnd];
チェックアウトNSScannerのドキュメントから選択する多くの方法のために。
isAtEnd
。
特定の文字列内のすべての文字が数値であることを確認する最も簡単な方法は、おそらく次のとおりです。
NSString *trimmedString = [newString stringByTrimmingCharactersInSet:[NSCharacterSet decimalDigitCharacterSet]];
if([trimmedString length])
{
NSLog(@"some characters outside of the decimal character set found");
}
else
{
NSLog(@"all characters were in the decimal character set");
}
許容可能な文字を完全に制御したい場合は、他のNSCharacterSetファクトリメソッドのいずれかを使用します。
この最初の質問はObjective-Cに関するものでしたが、Swiftが発表される何年も前に投稿されました。したがって、Googleから来て、Swiftを使用するソリューションを探している場合は、次のようにします。
let testString = "12345"
let badCharacters = NSCharacterSet.decimalDigitCharacterSet().invertedSet
if testString.rangeOfCharacterFromSet(badCharacters) == nil {
print("Test string was a number")
} else {
print("Test string contained non-digit characters.")
}
文字列が数字のみであることを確認する必要がある場合のSwift 3ソリューション:
CharacterSet.decimalDigits.isSuperset(of: CharacterSet(charactersIn: myString))
.inverted
アクションや他のアクションをしないことが好きです。
明確にするために、これは文字列の整数に対して機能します。
上記のジョンの答えに基づいた小さなヘルパーカテゴリがあります:
.hファイル
@interface NSString (NumberChecking)
+(bool)isNumber:(NSString *)string;
@end
.mファイル
#import "NSString+NumberChecking.h"
@implementation NSString (NumberChecking)
+(bool)isNumber {
if([self rangeOfCharacterFromSet:[[NSCharacterSet decimalDigitCharacterSet] invertedSet]].location == NSNotFound) {
return YES;
}else {
return NO;
}
}
@end
使用法:
#import "NSString+NumberChecking.h"
if([someString isNumber]) {
NSLog(@"is a number");
}else {
NSLog(@"not a number");
}
Swift 3ソリューションは次のようになります。
extension String {
var doubleValue:Double? {
return NumberFormatter().number(from:self)?.doubleValue
}
var integerValue:Int? {
return NumberFormatter().number(from:self)?.intValue
}
var isNumber:Bool {
get {
let badCharacters = NSCharacterSet.decimalDigits.inverted
return (self.rangeOfCharacter(from: badCharacters) == nil)
}
}
}
John Calsbeekの答えはほぼ正しいですが、一部のUnicodeエッジケースが省略されています。
のドキュメントによるとdecimalDigitCharacterSet
、そのセットには、Unicodeによってとして分類されたすべての文字が含まれていNd
ます。したがって、彼らの答えは、とりわけ以下を受け入れます:
१
(U + 0967 DEVANAGARI DIGIT ONE)᠑
(U + 1811 MONGOLIAN DIGIT ONE)𝟙
(U + 1D7D9数学ダブルストラク桁1)ある意味これは正しいですが、の各文字はNd
10進数字にマップされますが、ほとんどの場合、質問者が期待したものではありません。これを書いている時点では、610のコードポイントがとして分類されておりNd
、そのうちの10文字のみが0
(U + 0030)から9
(U + 0039)までの予想文字です。
この問題を修正するには、受け入れ可能な文字を正確に指定するだけです。
NSCharacterSet* notDigits =
[[NSCharacterSet characterSetWithCharactersInString:@"0123456789"] invertedSet];
if ([newString rangeOfCharacterFromSet:notDigits].location == NSNotFound)
{
// newString consists only of the digits 0 through 9
}
さらに別のオプション:
- (BOOL)isValidNumber:(NSString*)text regex:(NSString*)regex {
@try {
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];
return [predicate evaluateWithObject:text];
}
@catch (NSException *exception) {
assert(false);
return NO;
}
}
使用例:
BOOL isValid = [self isValidNumber:@"1234" regex:@"^[0-9]+$"];
@John Calsbeekの回答の拡張 、および@Jeffと@gyratoryサーカスのコメントの明確化。
+ (BOOL)doesContainDigitsOnly:(NSString *)string
{
NSCharacterSet *nonDigits = [[NSCharacterSet decimalDigitCharacterSet] invertedSet];
BOOL containsDigitsOnly = [string rangeOfCharacterFromSet:nonDigits].location == NSNotFound;
return containsDigitsOnly;
}
+ (BOOL)doesContainNonDigitsOnly:(NSString *)string
{
NSCharacterSet *digits = [NSCharacterSet decimalDigitCharacterSet];
BOOL containsNonDigitsOnly = [string rangeOfCharacterFromSet:digits].location == NSNotFound;
return containsNonDigitsOnly;
}
以下をカテゴリーメソッドとして追加できます NSString
- (BOOL)doesContainDigitsOnly
{
NSCharacterSet *nonDigits = [[NSCharacterSet decimalDigitCharacterSet] invertedSet];
BOOL containsDigitsOnly = [self rangeOfCharacterFromSet:nonDigits].location == NSNotFound;
return containsDigitsOnly;
}
- (BOOL)doesContainNonDigitsOnly
{
NSCharacterSet *digits = [NSCharacterSet decimalDigitCharacterSet];
BOOL containsNonDigitsOnly = [self rangeOfCharacterFromSet:digits].location == NSNotFound;
return containsNonDigitsOnly;
}
Swift 3の場合
var onlyDigits: CharacterSet = CharacterSet.decimalDigits.inverted
if testString.rangeOfCharacter(from: onlyDigits) == nil {
// String only consist digits 0-9
}
あなたがからある数字がある場合混合言語、その使用(またはしないでください)0-9桁形式を、あなたは、任意の数を探します正規表現を実行する必要があります、次のことはするすべての桁を変換することです0- 9形式(実際の値が必要な場合):
// Will look for any language digits
let regex = try NSRegularExpression(pattern: "[^[:digit:]]", options: .caseInsensitive)
let digitsString = regex.stringByReplacingMatches(in: string,
options: NSRegularExpression.MatchingOptions(rawValue: 0),
range: NSMakeRange(0, string.count), withTemplate: "")
// Converting the digits to be 0-9 format
let numberFormatter = NumberFormatter()
numberFormatter.locale = Locale(identifier: "EN")
let finalValue = numberFormatter.number(from: digitsString)
if let finalValue = finalValue {
let actualValue = finalValue.doubleValue
}
最も簡単で信頼性の高い方法はDouble
、結果がnil
- としてキャストしようとすることです。これを正当な数に形成することはできません。
let strings = ["test", "123", "123.2", "-123", "123-3", "123..22", ".02"]
let validNumbers = strings.compactMap(Double.init)
print(validNumbers)
// prints [123.0, 123.2, -123.0, 0.02]
ドキュメントの詳細:https : //developer.apple.com/documentation/swift/double/2926277-init
.
文字のみで構成されます