0

我在 C 中有以下功能:

char* AUDIO_GetFile(FIL* fil, DIR* dir, FILINFO* fno) {
    FRESULT res;
    res = f_readdir(dir, fno); /* Read a directory item */
    if (fno->fname[0] == '\0')} /* End of files */
       ...
    }
    ...
    return &(fno->fname);
}

但是我需要一个临时变量来保留旧值然后返回&(fno->fname)。我创建了局部变量FILINFO t_fno,然后将其传递给函数,res = f_readdir(dir, &t_fno);一切正常。

但是,如果我想使用 t_fno 作为本地指针,然后将其传递给 f_readdir,我不会得到实际数据,而是会在if (t_fno->fname[0] == '\0')}. 我也尝试过初始化FILINFO* t_fno = NULL,但这没有帮助。

char* AUDIO_GetFile(FIL* fil, DIR* dir, FILINFO* fno) {
    FRESULT res;
    FILINFO* t_fno;
    res = f_readdir(dir, t_fno); /* Read a directory item */
    if (t_fno->fname[0] == '\0')} /* crash */
       ...
    }
    ...
    fno = t_fno;
    return &(fno->fname);
}


FRESULT f_readdir (
    DIR* dp,            /* Pointer to the open directory object */
    FILINFO* fno        /* Pointer to file information to return */
)
{
    FRESULT res;
    DEFINE_NAMEBUF;


    res = validate(dp);                     /* Check validity of the object */
    if (res == FR_OK) {
        if (!fno) {
            res = dir_sdi(dp, 0);           /* Rewind the directory object */
        } else {
            INIT_BUF(*dp);
            res = dir_read(dp, 0);          /* Read an item */
            if (res == FR_NO_FILE) {        /* Reached end of directory */
                dp->sect = 0;
                res = FR_OK;
            }
            if (res == FR_OK) {             /* A valid entry is found */
                get_fileinfo(dp, fno);      /* Get the object information */
                res = dir_next(dp, 0);      /* Increment index for next */
                if (res == FR_NO_FILE) {
                    dp->sect = 0;
                    res = FR_OK;
                }
            }
            FREE_BUF();
        }
    }

    LEAVE_FF(dp->fs, res);
}
4

1 回答 1

1

根据其文档f_readdir()期望其第二个参数是“[p] 指向文件信息结构的指针,以存储有关读取项目的信息。” 这与您描述的第一个用途一致,其中您声明t_fno为本地FILINFO结构,并传递其地址。它与声明为指针、使其未初始化并传递其不确定的初始值不一致。t_fno

第二种方式会产生未定义的行为,这似乎非常合理,因为可能最慈善的解释是您已指示函数在某个随机位置写入数据。即使这样做成功,您也不知道会产生什么效果,或者您以后尝试读回它们时数据是否会保持不变,或者您是否可以稍后再读回它们。C 不要求或为您提供任何推理 - 它只是将行为声明为未定义。你不能以这种方式安全或可靠地完成工作。

正如您所观察到的,初始化指针没有帮助,除非(您没有探索)您将它初始化为指向与FILINFO结构兼容的实际存储。指针声明本身只提供指针,而不提供任何指向的东西。例子:

char* AUDIO_GetFile(FIL* fil, DIR* dir, FILINFO* fno) {
    FRESULT res;
    FILINFO t_info;
    FILINFO* t_fno = &t_info;

    res = f_readdir(dir, t_fno); /* Read a directory item */

    // ...

在大多数情况下,我宁愿不t_fno单独维护,但保留它可能有可行的理由。

于 2017-04-06T18:27:43.850 回答