3

我对 Xcode 非常陌生,并尝试(以百万计)解析 CSV 文件。我已经阅读了很多贡献并且我正在管理它,但是当我的 NSScanner 截获一个空字段时我遇到了问题:“Field_A,Field_B,,Field_D”。我猜这是因为它默认忽略空白空间,或者在这种情况下根本没有空间。

字符串是:

"个人","2011-01-01","个人","香烟",,4.60,"现金","",

我尝试使用scanLocation对其进行调试:

2011-04-22 15:57:32.414 Spending[42015:a0f] Before while...scan location is:0
2011-04-22 15:57:32.414 Spending[42015:a0f] Account: "Personal" - scan location is:10
2011-04-22 15:57:32.415 Spending[42015:a0f] Date: "2011-01-01" - scan location is:23
2011-04-22 15:57:32.415 Spending[42015:a0f] Category: "Personal" - scan location is:34
2011-04-22 15:57:32.416 Spending[42015:a0f] Subcategory: "Cigarettes" - scan location is:47
2011-04-22 15:57:32.416 Spending[42015:a0f] Income: 4.600000 - scan location is:53
2011-04-22 15:57:32.416 Spending[42015:a0f] Expense: 0.000000 - scan location is:53
2011-04-22 15:57:32.417 Spending[42015:a0f] Payment: "Cash" - scan location is:60
2011-04-22 15:57:32.417 Spending[42015:a0f] Note: "" - scan location is:63

正如您所看到的那样,即使费用字段也没有任何价值(应该是 4.60)。

这是相关的代码:

NSScanner *scanner = [NSScanner scannerWithString:fileString];
    [scanner setCharactersToBeSkipped: [NSCharacterSet characterSetWithCharactersInString:@"\n, "]];

    NSString *account, *date, *category, *subcategory, *payment, *note;
    float income, expense;

    // Set up data delimiter using comma
    NSCharacterSet *commaSet;
    commaSet = [NSCharacterSet characterSetWithCharactersInString:@","];

    NSLog (@"Before while...scan location is:%d\n", scanner.scanLocation);

    [scanner scanUpToCharactersFromSet:commaSet intoString:&account];
    NSLog(@"Account: %@ - scan location is:%d\n",account, scanner.scanLocation);

    [scanner scanUpToCharactersFromSet:commaSet intoString:&date];
    NSLog(@"Date: %@ - scan location is:%d\n",date, scanner.scanLocation);

    [scanner scanUpToCharactersFromSet:commaSet intoString:&category]; 
    NSLog(@"Category: %@ - scan location is:%d\n",category, scanner.scanLocation);

    [scanner scanUpToCharactersFromSet:commaSet intoString:&subcategory]; 
    NSLog(@"Subcategory: %@ - scan location is:%d\n",subcategory, scanner.scanLocation);

    [scanner scanFloat:&income];
    NSLog(@"Income: %f - scan location is:%d\n",income, scanner.scanLocation);

    [scanner scanFloat:&expense]; 
    NSLog(@"Expense: %f - scan location is:%d\n",expense, scanner.scanLocation);

    [scanner scanUpToCharactersFromSet:commaSet intoString:&payment]; 
    NSLog(@"Payment: %@ - scan location is:%d\n",payment, scanner.scanLocation);

    [scanner scanUpToCharactersFromSet:commaSet intoString:&note];
    NSLog(@"Note: %@\n - scan location is:%d",note, scanner.scanLocation);

我尝试通过 NSScanner 类参考仔细查看,但无法理解?你有什么?

谢谢,法布里齐奥。

4

3 回答 3

2

在 Objective-C 中解析 CSV?这听起来很熟悉:

https://github.com/davedelong/CHCSVParser

免责声明:我写的。:)


对于您正在做的事情,您可以只获取文件并通过类似+[NSArray arrayWithContentsOfCSVFile:encoding:error:]方法的方式运行,或者您可以将其读入字符串并执行以下操作:

NSString *csv = @"\"Personal\",\"2011-01-01\",\"Personal\",\"Cigarettes\",,4.60,\"Cash\",\"\",";
NSLog(@"%@", [csv CSVComponents]);

哪些日志:

2011-04-22 09:51:16.651 CHCSVParser[2658:903] (
        (
        Personal,
        "2011-01-01",
        Personal,
        Cigarettes,
        "",
        "4.60",
        Cash,
        ""
    )
)

(请注意,这是一个NSArrayof NSArraysNSStrings

如果您担心内存堆积,那么您也可以CHCSVParser直接使用 a 并通过委托接收信息。它的运作方式与运作方式几乎相同NSXMLParser

于 2011-04-22T16:49:43.920 回答
0

看看这篇关于CSV 扫描仪的文章。

这是另一篇文章

Dave DeLong 的解决方案也很有效。

底线:CSV 似乎微不足道,但如果您想优雅地处理任何扔给您的 CSV,它确实不是。

于 2011-04-22T16:01:41.200 回答
0

扫描仪看不到您的空白字段的原因是您告诉它跳过逗号。您使用一组 3 个字符调用了 setCharactersToBeSkipped:

  1. '\n' 换行符
  2. ',' 逗号
  3. ' ' 空间

然后,当您要求扫描仪“scanFloat”时,它会遍历任何可跳过的字符,直到达到十进制数。这就是跳过空字段的方式。

如果要捕获空字段,请从字符集中删除逗号以跳过。然后,任何时候扫描函数找到一个空字段,它都会返回 NO。发生这种情况时,您可能必须手动增加扫描位置。

于 2012-07-08T03:11:09.903 回答