8

我需要最有效地在有限宽度的区域中打印一个浮点值。我使用的是NSNumberFormatter,并且我将小数点后的两个数字设置为默认值,这样当我有一个类似的数字时,234.25它会按原样打印:234.25. 但是当我1234.25有时,我希望它被打印为:1234.3,并且 11234.25应该被打印11234

我需要在点之后最多两位数,如果我在点之后有数字,则总共最多需要五位数字,但如果整数部分有更多,它也应该打印超过五位数字。

我看不到限制NSNumberFormatter. 这是否意味着我应该编写自己的函数来以这种方式格式化数字?如果是这样,那么获取数字的整数和小数部分的位数的正确方法是什么?我也更喜欢使用CGFLoat,而不是NSNumber避免额外的类型转换。

4

4 回答 4

15

您正在寻找“最大有效数字”和“最大小数位数”的组合,以及特定的舍入行为。NSNumberFormatter等于任务:

float twofortythreetwentyfive = 234.25;
float onetwothreefourtwentyfive = 1234.25;
float eleventwothreefourtwentyfive = 11234.25;

NSNumberFormatter * formatter =  [[NSNumberFormatter alloc] init];
[formatter setUsesSignificantDigits:YES];
[formatter setMaximumSignificantDigits:5];
[formatter setMaximumFractionDigits:2];
[formatter setRoundingMode:NSNumberFormatterRoundCeiling];

NSLog(@"%@", [formatter stringFromNumber:[NSNumber numberWithFloat:twofortythreetwentyfive]]);
NSLog(@"%@", [formatter stringFromNumber:[NSNumber numberWithFloat:onetwothreefourtwentyfive]]);
NSLog(@"%@", [formatter stringFromNumber:[NSNumber numberWithFloat:eleventwothreefourtwentyfive]]);

结果:

2012-04-26 16:32:04.481 有效数字[11565:707] 234.25
2012-04-26 16:32:04.482 有效数字[11565:707] 1234.3
2012-04-26 16:32:04.483 有效数字[11565:707] 11235

于 2012-04-26T16:29:00.620 回答
1

代码 :

#define INTPARTSTR(X) [NSString stringWithFormat:@"%d",(int)X]
#define DECPARTSTR(X) [NSString stringWithFormat:@"%d",(int)(((float)X-(int)X)*100)]

- (NSString*)formatFloat:(float)f
{
    NSString* result;

    result = [NSString stringWithFormat:@"%.2f",f];

    if ([DECPARTSTR(f) isEqualToString:@"0"]) return INTPARTSTR(f);
    if ([INTPARTSTR(f) length]==5) return INTPARTSTR(f);

    if ([result length]>5)
    {
        int diff = (int)[result length]-7;

        NSString* newResult = @"";

        for (int i=0; i<[result length]-diff-1; i++)
            newResult = [newResult stringByAppendingFormat:@"%c",[result characterAtIndex:i]];

        return newResult;
    }


    return result;
}

测试它:

- (void)awakeFromNib
{
    NSLog(@"%@",[self formatFloat:234.63]);
    NSLog(@"%@",[self formatFloat:1234.65]);
    NSLog(@"%@",[self formatFloat:11234.65]);
    NSLog(@"%@",[self formatFloat:11234]);
}

输出 :

2012-04-26 19:27:24.429 newProj[1798:903] 234.63
2012-04-26 19:27:24.432 newProj[1798:903] 1234.6
2012-04-26 19:27:24.432 newProj[1798:903] 11234
2012-04-26 19:27:24.432 newProj[1798:903] 11234
于 2012-04-26T16:28:58.770 回答
0

这是我在代码中实现这一点的方式。不知道效率如何,希望不会差。

所以我创建了一个全局 NSNumberFormatter

NSNumberFormatter* numFormatter; 

并在某处初始化它:

numFormatter=[[NSNumberFormatter alloc]init];

然后我使用以下函数格式化数字:

- (NSString*)formatFloat:(Float32)number withOptimalDigits:(UInt8)optimalDigits maxDecimals:(UInt8)maxDecimals
{
    NSString* result;
    UInt8 intDigits=(int)log10f(number)+1;
    NSLog(@"Formatting %.5f with maxDig: %d maxDec: %d intLength: %d",number,optimalDigits,maxDecimals,intDigits);  

    numFormatter.maximumFractionDigits=maxDecimals;
    if(intDigits>=optimalDigitis-maxDecimals) {
        numFormatter.usesSignificantDigits=YES;
        numFormatter.maximumSignificantDigits=(intDigits>optimalDigits)?intDigits:optimalDigits;
    } else {
        numFormatter.usesSignificantDigits=NO;
    }
    result = [numFormatter stringFromNumber:[NSNumber numberWithFloat:number]];

    return result;
}
于 2012-04-28T17:07:02.820 回答
-1

在 iOS 8上使用maximumFractionDigitsmaximumSignificantDigits一起使用时这是一个错误吗?NSNumberForamtter

        NSNumberFormatter *formatter = [[NSNumberFormatter alloc] init];
        formatter.maximumFractionDigits = 2;  
        formatter.maximumSignificantDigits = 3;
        NSLog(@"%@", [formatter stringFromNumber:@(0.3333)]); // output 0.333 expected 0.33

如果我只使用它就可以正常工作maximumFractionDigits

        NSNumberFormatter *formatter = [[NSNumberFormatter alloc] init];
        formatter.maximumFractionDigits = 2;
        NSLog(@"%@", [formatter stringFromNumber:@(0.3333)]); // output expected .33

NSNumberFormatter maximumFractionDigits 和 maximumSignificantDigits 错误

于 2014-12-25T05:12:17.767 回答