Risposte:
Ecco un modo che non si basa sulla precisione limitata del tentativo di analizzare la stringa come numero:
NSCharacterSet* notDigits = [[NSCharacterSet decimalDigitCharacterSet] invertedSet];
if ([newString rangeOfCharacterFromSet:notDigits].location == NSNotFound)
{
// newString consists only of the digits 0 through 9
}
Vedi +[NSCharacterSet decimalDigitCharacterSet]e -[NSString rangeOfCharacterFromSet:].
[[NSCharacterSet characterSetWithCharactersInString:@"0123456789."] invertedSet]]
invertedSetstia restituendo una classe che avvolge il set in cui è stato creato e restituisce il valore opposto per tutte le query. Ma non lo so per certo.
Suggerirei di utilizzare il numberFromString:metodo della classe NSNumberFormatter , poiché se il numero non fosse valido, restituirà nil; in caso contrario, ti restituirà un NSNumber.
NSNumberFormatter *nf = [[[NSNumberFormatter alloc] init] autorelease];
BOOL isDecimal = [nf numberFromString:newString] != nil;
NOtua stringa. Vale anche la pena sapere: tornano anche YESper i numeri riempiti con spazi bianchi. A proposito, ho appena aggiunto un frammento di codice effettivo alla tua risposta, spero non ti dispiaccia.
Convalida tramite espressione regolare, modello "^[0-9]+$", con il seguente metodo -validateString:withPattern:.
[self validateString:"12345" withPattern:"^[0-9]+$"];
"^[0-9]+(.{1}[0-9]+)?$"".".
"^[0-9]{4}$"."."e la lunghezza è compresa tra 2 e 5
"^[0-9]{2,5}$"."^-?\d+$"L'espressione regolare può essere verificata nel sito web in linea .
La funzione di supporto è la seguente.
// 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;
}
Prova nel parco giochi.
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 }E ho ricevuto un errorePlayground execution aborted: error: Execution was interrupted, reason: EXC_BAD_INSTRUCTION (code=EXC_I386_INVOP, subcode=0x0).
return matchRange.location != NSNotFound;
^-?\d+$, l'ho verificato sul sito: regex101.com
Puoi creare un NSScanner e scansionare semplicemente la stringa:
NSDecimal decimalValue;
NSScanner *sc = [NSScanner scannerWithString:newString];
[sc scanDecimal:&decimalValue];
BOOL isDecimal = [sc isAtEnd];
Consulta la documentazione di NSScanner per ulteriori metodi tra cui scegliere.
isAtEnd.
Penso che il modo più semplice per verificare che ogni carattere all'interno di una determinata stringa sia numerico è probabilmente:
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");
}
Utilizzare uno degli altri metodi predefiniti NSCharacterSet se si desidera il controllo completo sui caratteri accettabili.
Questa domanda originale riguardava Objective-C, ma è stata anche pubblicata anni prima dell'annuncio di Swift. Quindi, se vieni qui da Google e stai cercando una soluzione che utilizzi Swift, ecco qui:
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.")
}
Soluzione Swift 3 se è necessario verificare che la stringa contenga solo cifre:
CharacterSet.decimalDigits.isSuperset(of: CharacterSet(charactersIn: myString))
.invertedazioni inutili e di altro tipo.
per essere chiari, questo funziona per gli interi nelle stringhe.
ecco una piccola categoria di aiuto basata sulla risposta di John sopra:
nel file .h
@interface NSString (NumberChecking)
+(bool)isNumber:(NSString *)string;
@end
nel file .m
#import "NSString+NumberChecking.h"
@implementation NSString (NumberChecking)
+(bool)isNumber {
if([self rangeOfCharacterFromSet:[[NSCharacterSet decimalDigitCharacterSet] invertedSet]].location == NSNotFound) {
return YES;
}else {
return NO;
}
}
@end
utilizzo:
#import "NSString+NumberChecking.h"
if([someString isNumber]) {
NSLog(@"is a number");
}else {
NSLog(@"not a number");
}
La soluzione Swift 3 potrebbe essere come:
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)
}
}
}
La risposta di John Calsbeek è quasi corretta ma omette alcuni casi limite Unicode.
Secondo la documentazione didecimalDigitCharacterSet , quel set include tutti i caratteri classificati da Unicode come Nd. Così la loro risposta accetterà, tra gli altri:
१ (U + 0967 DEVANAGARI DIGIT ONE)᠑ (U + 1811 MONGOLIAN DIGIT ONE)𝟙 (U + 1D7D9 MATEMATICA DOUBLE-STRUCK DIGIT ONE)Anche se in un certo senso questo è corretto - ogni carattere in Ndviene mappato su una cifra decimale - quasi certamente non è quello che il richiedente si aspettava. Al momento della stesura di questo documento ci sono 610 punti di codice classificati comeNd , solo dieci dei quali sono i caratteri previsti 0(U + 0030) fino a 9(U + 0039).
Per risolvere il problema, è sufficiente specificare esattamente i caratteri accettabili:
NSCharacterSet* notDigits =
[[NSCharacterSet characterSetWithCharactersInString:@"0123456789"] invertedSet];
if ([newString rangeOfCharacterFromSet:notDigits].location == NSNotFound)
{
// newString consists only of the digits 0 through 9
}
Ancora un'altra opzione:
- (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;
}
}
Esempio di utilizzo:
BOOL isValid = [self isValidNumber:@"1234" regex:@"^[0-9]+$"];
Un'estensione della @ John Calsbeek 'risposta s, e chiarire @Jeff e circo @gyratory ' s commenti.
+ (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;
}
Quanto segue può essere aggiunto come metodi di categoria per 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;
}
Verifica se una stringa è un numero Potrebbe essere utile
int i = [@"12.3" rangeOfCharacterFromSet: [ [NSCharacterSet characterSetWithCharactersInString:@"0123456789."] invertedSet] ].location;
if (i == NSNotFound) {
//is a number
}
Estensione rapida:
extension NSString {
func isNumString() -> Bool {
let numbers = NSCharacterSet(charactersInString: "0123456789.").invertedSet
let range = self.rangeOfCharacterFromSet(numbers).location
if range == NSNotFound {
return true
}
return false
} }
Per Swift 3
var onlyDigits: CharacterSet = CharacterSet.decimalDigits.inverted
if testString.rangeOfCharacter(from: onlyDigits) == nil {
// String only consist digits 0-9
}
Quando hai cifre che provengono da lingue miste , che usano (o non usano) i formati 0-9 cifre, dovrai eseguire una regex che cercherà qualsiasi numero, la prossima cosa è convertire tutte le cifre in 0- Formato 9 (se è necessario il valore effettivo):
// 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
}
Il modo più semplice e affidabile è provare a lanciare come Double, se il risultato è nil- non può essere formato in un numero legittimo.
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]
Maggiori informazioni nella documentazione: https://developer.apple.com/documentation/swift/double/2926277-init
.carattere