0

我在 OS X 10.8 上使用 Xcode 4.5.1 创建了一个项目。配置为: 项目:架构:标准(32/64 位英特尔) 基础 SDK:OS X 10.7 部署目标:10.7

起初一切都很顺利。但是,当我在 OS X 10.7 上将此项目复制到另一台带有 Xcode 4.4.1 的 Macbook 时,会引发运行时错误。


上面的配置未更改,并且从文件中获取配置字符串的方法发生错误:

+(NSString *)configValueFromFile:(NSString *)path
               withParameter:(NSString *)parameter
{
const char *cPath = [path UTF8String];
const char *cPara = [[parameter uppercaseString] UTF8String];
char buff[CFG_CONFIG_READ_TOOL_BUFFER_LEN];
size_t paraLen = [parameter length];
size_t lineLen;
FILE *fileFd = NULL;
//struct stat dummyFileStat;
BOOL isValueFound = NO;

NSString *ret = nil;

/* open file *//*
if (0 != stat(cPath, &dummyFileStat))
{
    NSLog(@"stat() %@: %@", path, [AMCTools strError]);
    return nil;
}*/


fileFd = fopen(cPath, "r");
if(!fileFd)
{
    NSLog(@"fopen() %@: %@", path, [AMCTools strError]);
    return nil;
}

/* read configuration */
while((!feof(fileFd)) && (!isValueFound))
{
    @autoreleasepool {
        size_t tmp;

        fgets(buff, sizeof(buff), fileFd);
        if ('\0' == buff[sizeof(buff) - 1])
        {
            break;
        }

        /* get a line */
        lineLen = strlen(buff);
        if ((0 == lineLen) || ('#' == buff[0]))
        {
            continue;   /* !!!!!!! */
        }

        while(('\n' == buff[lineLen - 1]) ||
              ('\r' == buff[lineLen - 1]))
        {
            buff[lineLen - 1] = '\0';
            lineLen --;
        }

        //NSLog(@"Get line: %s", buff);

        /* upper case the parameter */
        for (tmp = 0;
             (tmp < lineLen) || ('=' != buff[tmp]);
             tmp++)
        {
            if ((buff[tmp] >= 'a') && (buff[tmp] <= 'z'))
            {
                buff[tmp] += 'A' - 'a';
            }
        }

        /* compare */
        if (0 == strncmp(cPara, buff, paraLen))
        {
            isValueFound = YES;

            /* fetch value sector */
            for (tmp = paraLen; tmp < lineLen; tmp++)
            {
                if ('=' == buff[tmp])
                {
                    tmp++;
                    break;
                }
            }   // end: for (tmp = paraLen...)

            if (tmp >= lineLen)
            {
                /* no parameter values */
                ret = @"";
            }
            else
            {
                /* skip blank */
                for (/**/; tmp < lineLen; tmp++)
                {
                    if ((' ' != buff[tmp]) &&
                        ('\t' != buff[tmp]))
                    {
                        break;
                    }
                }
                ret = [NSString stringWithUTF8String:buff+tmp];
            }
        }   // end: compare
    }   // end: autoreleasepool
}   //end: while(...)


ENDS:
    if (fileFd)
    {
        fclose(fileFd);
    }
    return ret;
}

这个方法一直很好,直到“return ret”,它抛出一个错误“线程1:信号SIGABRT ”;


我该怎么办?

谢谢!!


科迪的更多信息:

  1. 关于您的担忧:

哦,我忘了提到我确实使用了 ARC。它不应该被释放。

你提到了这条线:

const char *cPara = [[parameter uppercaseString] UTF8String];

如果不合适,我应该如何更正?

4

1 回答 1

0

嗯,看起来你的返回值被释放了。据我所知stringWithUTF8String:返回自动释放的字符串。在你的情况下,它在@autoreleasepool块内。当块结束时 ret 被释放并且你返回一些垃圾。

我也担心这条线:

const char *cPara = [[parameter uppercaseString] UTF8String];

这里 cPara 将指向临时字符串内。这可能是泄漏,也可能是崩溃(使用 ARC)。您应该将大写字符串存储在一些中间变量中:

NSString uppercaseParam = [parameter uppercaseString];
const char* cPara = [uppercaseParam UTF8String];


你绝对应该看看 ARC - 这是一个非常方便的功能。

于 2013-04-11T06:37:04.553 回答