1

我正在尝试创建一个函数,该函数附加!!在文本文件的(包含)到(不包含)每一行的末尾。在我遇到了几次并发症后,我实际上成功了。但也不完全。


int incldue_auto (char *script, unsigned int offsetLine, unsigned int endLine)
{
    size_t  fileSize;
    size_t  content = (endLine - offsetLine) * 3; // New characters
    char    *buffer;
    FILE*   fp;

    if((fp = fopen(script, "r")) == NULL)                               //return(1);
    if(fseek(fp, 0l, SEEK_END) != 0)                                    //return(2);
    if((fileSize = ftell(fp)) == (-1l))                                 //return(3);
    if(fseek(fp, 0l, SEEK_SET) != 0)                                    //return(2);
    if((buffer = calloc(fileSize + content, sizeof(char))) == NULL)     //return(4);
    if(fread(buffer, sizeof(char), fileSize, fp) != fileSize)           //return(5);
    if(fclose(fp) == EOF)                                               //return(6);
    {
        int i, i2;
        int lines = 0, ln = 0;
        for(i = 0; i < fileSize; i++)
        {
            if(ln >= (endLine - offsetLine) || i[buffer] == '\0') break;
            if(i[buffer] == '\n')
            {
                lines++;

                if(lines >= offsetLine && lines < endLine)
                {
                    char* p = (buffer + i); // \n

                    //if(*(p - 1) == '\n') continue;

                    memmove(p + 3,
                            p,
                            strlen(p)); // <-- Problematic line (I think)

                    memcpy(p, " !!", 3);

                    i += 3;
                    ln++;
                }
            }
        }

        fp = fopen(script, "w");
        fwrite(buffer, fileSize + content, sizeof(char), fp);
        fclose(fp);
    }
    free(buffer);
    return 0;
}

它相对工作得很好,除了它没有附加到最后一行。它最后用空格(可能是NULL)填充文本文件。我认为这是因为我也在移动 enzeroed 附加区域content

memmove(p + 3,
        p,
        strlen(p)); // <-- Problematic line (I think)

在此处输入图像描述 所以也许我需要弄清楚我必须使用什么合适的公式才能完成这项工作。关于如何完成这项工作的任何想法?

4

2 回答 2

2

您的循环条件错误:

for(i = 0; i < fileSize; i++)

在您将内容附加到某些行之后,最后一行的末尾将被移出filesize,因此将不再被处理。

您使用 测试是否已到达缓冲区的末尾i[buffer] == '\0',但这可能会导致在完成所有班次后读取缓冲区之外的内容,并且有更多的尾随零。这不应该发生,因为首先检查结束行条件,但最好再calloc输入一个字符以确保安全,这将buffer生成一个以零结尾的字符串。

当你移动的行数少于请求的行数时,因为文件没有请求的行数——比如你在一个 30 行的文件中将第 20 行移动到 40 行——你仍然会打印出尾随的零。您可以将零终止缓冲区打印为字符串,也可以跟踪附加的实际数据量。

您保留两个冗余行数,lines并且ln. 选择一个并删除另一个;它只会不必要地混淆代码。

于 2015-08-23T08:40:02.283 回答
0

也许您可以在循环中再添加一个条件:

if(lines >= offsetLine && lines < endLine)
{
    ...//your code here before 'memcpy(p, " !!", 3);'
    if(lines==endline)
    {
        memmove(p + 3, p, strlen(p));    
    }
    //the end of your code here:
    memcpy(p, " !!", 3);//your code here
    i += 3;
    ln++;
}

如果这是最后一行,那么您将再移动 3 个字符并输入 '!!' 就在这之后。

于 2015-08-23T08:00:55.300 回答