Как преобразовать NSString в NSNumber


569

Как я могу преобразовать , NSStringсодержащий ряд любого примитивного типа данных (например int, float, char, unsigned intи т.д.)? Проблема в том, что я не знаю, какой тип числа будет содержать строка во время выполнения.

У меня есть идея, как это сделать, но я не уверен, что это работает с любым типом, а также со значениями без знака и с плавающей точкой:

long long scannedNumber;
NSScanner *scanner = [NSScanner scannerWithString:aString];
[scanner scanLongLong:&scannedNumber]; 
NSNumber *number = [NSNumber numberWithLongLong: scannedNumber];

Спасибо за помощь.

Ответы:


1245

Используйте NSNumberFormatter:

NSNumberFormatter *f = [[NSNumberFormatter alloc] init];
f.numberStyle = NSNumberFormatterDecimalStyle;
NSNumber *myNumber = [f numberFromString:@"42"];

Если строка не является допустимым числом, то myNumberбудет nil. Если это действительное число, то теперь у вас есть все возможности, NSNumberчтобы выяснить, что это за число на самом деле.


16
Для людей, где это не работает: проверьте, связано ли это с вашей локалью. NSNumberFormatter (насколько я знаю) по умолчанию использует локаль США, т.е. ожидает, что десятичный разделитель будет "." персонаж. Если вы используете «,» для разделения дроби, вам может потребоваться указать форматировщику использовать ваш текущий языковой стандарт: [f setLocale: [NSLocale currentLocale]];
Pille

2
@pille по умолчанию это локаль +currentLocale, но вы правы; всегда нужно учитывать локаль при работе с конвертируемой информацией в удобочитаемую форму.
Дейв Делонг

Где вы обнаружили, что по умолчанию это currentLocale? Я могу подтвердить, что numberFromString использует точечную запись даже на французском телефоне, где используется запятая. Я могу подтвердить, что по умолчанию, когда вы
выводите

Но что, если строка может содержать либо int, либо float?
GeneCode

179

Вы можете использовать -[NSString integerValue], -[NSString floatValue]и т.д. Тем не менее, правильный ( с учетом локал и т.д.) способ сделать это состоит в использовании , -[NSNumberFormatter numberFromString:]который даст вам NSNumber преобразованных из соответствующей местности и с учетом настроек NSNumberFormatter( в том числе , будет ли это позволить плавающему баллы).


27
+ q В зависимости от ситуации, не зависящий от локали может быть правильным способом.
Тило

2
Я должен был преобразовать @"2000"в intи [@"2000" integerValue]работал хорошо, и немного проще для моего случая.
Беси

2
Существуют также огромные различия в производительности среди этих методов.
Том Андерсен

2
это не работает с ARC: оно не преобразует NSInteger в NSNumber. Я должен в дальнейшем использовать[NSNumber numberWithInteger ...]
ученик

Что @Thilo сказал. «Правильный» и «чувствительный к локали» не являются синонимами. Правильная вещь зависит от того, как число попало в строку. Если вы читаете его из пользовательского ввода, то вам нужно локально-чувствительное. Из любого другого источника (т.е. вы помещаете его туда программно, вы потребляете его как часть ответа от какого-то внешнего API и т. Д.), Локально-чувствительный не подходит и может даже давать неверные результаты.
Аромат

138

Objective-C

(Примечание: этот метод не подходит для разных локалей, но немного быстрее, чем a NSNumberFormatter)

NSNumber *num1 = @([@"42" intValue]);
NSNumber *num2 = @([@"42.42" floatValue]);

стриж

Простой, но грязный способ

// Swift 1.2
if let intValue = "42".toInt() {
    let number1 = NSNumber(integer:intValue)
}

// Swift 2.0
let number2 = Int("42')

// Swift 3.0
NSDecimalNumber(string: "42.42") 

// Using NSNumber
let number3 = NSNumber(float:("42.42" as NSString).floatValue)

Способ расширения Это лучше, на самом деле, потому что он будет хорошо работать с локалями и десятичными числами.

extension String {

    var numberValue:NSNumber? {
        let formatter = NSNumberFormatter()
        formatter.numberStyle = .DecimalStyle
        return formatter.number(from: self)
    }
}

Теперь вы можете просто сделать:

let someFloat = "42.42".numberValue
let someInt = "42".numberValue

1
Я предпочитаю это на основе профилирования, когда я анализировал большое количество строковых значений. Использование этого синтаксиса вместо NSNumberFormatter привело к значительному сокращению времени, затрачиваемого на разбор строки в NSNumber. И да, NSNumberFormatter был кэширован и использован повторно.
androider

3
этот буквальный синтаксис не существовал, когда был задан этот вопрос. Я бы сказал, что сейчас это правильный ответ.
Крис

1
Я согласен, но учтите, что некоторые другие страны используют ,и .наоборот (например 42.000,42. Что-то, floatValueвероятно, не учитывается?
Кевин Р.

1
Это самое хорошее решение, но я только что провел тест, и вы потеряете точность без знака, поэтому СМОТРЕТЬ !!!
Stoff81

В моем случае использования мне нужно было получить ключ словаря NSNumber из строки NSString, поэтому @([@"42" intValue])работал отлично. Не беспокоиться о Locale.
Чак Круцингер

86

Для строк , начинающихся с целыми числами, например, @"123", @"456 ft", @"7.89"и т.д., использование -[NSString integerValue].

Итак, @([@"12.8 lbs" integerValue])это как делать [NSNumber numberWithInteger:12].


50

Вы также можете сделать это:

NSNumber *number = @([dictionary[@"id"] intValue]]);

Радоваться, веселиться!


26

Если вы знаете, что получаете целые числа, вы можете использовать:

NSString* val = @"12";
[NSNumber numberWithInt:[val intValue]];

11

Вот рабочий пример NSNumberFormatter, читающий локализованный номер NSString (xCode 3.2.4, osX 10.6), чтобы сэкономить другим часы, которые я только что потратил. Осторожно: хотя он может обрабатывать конечные пробелы (работает "8,765.4"), он не может обрабатывать начальные пробелы и не может обрабатывать произвольные текстовые символы. (Неверные строки ввода: «8» и «8q» и «8 q».)

NSString *tempStr = @"8,765.4";  
     // localization allows other thousands separators, also.
NSNumberFormatter * myNumFormatter = [[NSNumberFormatter alloc] init];
[myNumFormatter setLocale:[NSLocale currentLocale]]; // happen by default?
[myNumFormatter setFormatterBehavior:NSNumberFormatterBehavior10_4];
     // next line is very important!
[myNumFormatter setNumberStyle:NSNumberFormatterDecimalStyle]; // crucial

NSNumber *tempNum = [myNumFormatter numberFromString:tempStr];
NSLog(@"string '%@' gives NSNumber '%@' with intValue '%i'", 
    tempStr, tempNum, [tempNum intValue]);
[myNumFormatter release];  // good citizen

это не работает для меня. Я разрабатываю для 5.0 с xcode 4.3.2. есть идеи почему?
acecapades

6

Я хотел преобразовать строку в двойную. Этот ответ выше не работал для меня. Но это сделало: Как сделать преобразование строк в Objective-C?

Все, что я в значительной степени сделал, было:

double myDouble = [myString doubleValue];

Это не конвертирует NSInteger в NSNumber. Он преобразует его в двойной.
FabKremer

5

Спасибо всем! Я объединил обратную связь и, наконец, мне удалось преобразовать текстовый ввод (строку) в Integer. Кроме того, он может сказать мне, является ли ввод целым числом :)

NSNumberFormatter * f = [[NSNumberFormatter alloc] init];
[f setNumberStyle:NSNumberFormatterDecimalStyle];
NSNumber * myNumber = [f numberFromString:thresholdInput.text];

int minThreshold = [myNumber intValue]; 

NSLog(@"Setting for minThreshold %i", minThreshold);

if ((int)minThreshold < 1 )
{
    NSLog(@"Not a number");
}
else
{
    NSLog(@"Setting for integer minThreshold %i", minThreshold);
}
[f release];

4

Я думаю, что NSDecimalNumber сделает это:

Пример:

NSNumber *theNumber = [NSDecimalNumber decimalNumberWithString:[stringVariable text]]];

NSDecimalNumber является подклассом NSNumber, поэтому неявное приведение разрешено.


2

Как насчет стандарта С atoi?

int num = atoi([scannedNumber cStringUsingEncoding:NSUTF8StringEncoding]);

Как вы думаете, есть какие-то предостережения?


2

Вы можете просто использовать [string intValue]или [string floatValue]или[string doubleValue] т. Д.

Вы также можете использовать NSNumberFormatterкласс:


2

Вы также можете сделать, как этот код 8.3.3 IOS 10.3 Поддержка

[NSNumber numberWithInt:[@"put your string here" intValue]]



0

Я знаю, что это очень поздно, но приведенный ниже код работает для меня.

Попробуйте этот код

NSNumber *number = @([dictionary[@"keyValue"] intValue]]);

Это может помочь вам. Спасибо


0

Попробуй это

NSNumber *yourNumber = [NSNumber numberWithLongLong:[yourString longLongValue]];

Примечание - я использовал longLongValue согласно моему требованию. Вы также можете использовать integerValue, longValue или любой другой формат в зависимости от ваших требований.


-1
extension String {

    var numberValue:NSNumber? {
        let formatter = NumberFormatter()
        formatter.numberStyle = .decimal
        return formatter.number(from: self)
    }
}

let someFloat = "12.34".numberValue
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.