1

我试图用空格和引号来标记一个单词,但是在正确的输出之后我得到了一个奇怪的 malloc 错误。

我希望这个函数接受类似的东西:

hello world "SOme quote"

并且输出应该是:

hello
world
"some quote"

或者如果输入是:

hello world no quote 

输出应该是:

hello
world
no
quote

但是现在,我得到:

Hello
WOrld
"Hello World"
*** glibc detected *** ./a.out: free(): invalid next size (fast): 0x0000000001760010 ***
a.out: malloc.c:2451: sYSMALLOc: Assertion `(old_top == (((mbinptr) (((char *) &((av)->bins[((1) - 1) * 2])) - __builtin_offsetof (struct malloc_chunk, fd)))) && old_size == 0) || ((unsigned long) (old_size) >= (unsigned long)((((__builtin_offsetof (struct malloc_chunk, fd_nextsize))+((2 * (sizeof(size_t))) - 1)) & ~((2 * (sizeof(size_t))) - 1))) && ((old_top)->size & 0x1) && ((unsigned long)old_end & pagemask) == 0)' failed.
Aborted (core dumped)

看起来输出是正确的,但是之后它就搞砸了

代码是:

int process_command(char command[80]){
char curr_char;
char *word;
int start_pos;
int i;
int len;
len = strlen(command);
for(i=0,start_pos=0;i<strlen(command);i++){
    curr_char = command[i];
    if (curr_char == ' '){
        if (command[i-1]==' ') {start_pos++;continue;}
        word = malloc(i-start_pos*(sizeof(char)));
        strncpy(word,command+start_pos,i-start_pos);
        printf("%s\n",word);
        free(word);

        start_pos =i+1;

    }
    else if (curr_char == '\"'){
        word= malloc(len-i*(sizeof(char)));
        strncpy(word,command+i,len);
        printf("%s\n",word);
        free(word);
        i=len+len;
    }

}

return 0;
}
int main(){
    char buffer[80] = "Hello   WOrld  \"Hello World\"";
    process_command(buffer);
    return 0;
}

问题已解决!谢谢继承人更新的代码:

int process_command(char command[80]){
char curr_char;
char *word;
int start_pos;
int i;
int len;
int quote=0;
len = strlen(command);
for(i=0,start_pos=0;i<strlen(command);i++){
    curr_char = command[i];
    if (curr_char == ' '){      /*If there was a space found copy the stuff before the space*/
        if ( i>0 && command[i-1]==' ') {
            start_pos++;
            continue;
        }
        word = malloc(i-start_pos+1*(sizeof(char)));
        strncpy(word,command+start_pos,i-start_pos);
        word[i-start_pos+1]='\0';
        printf("%s\n",word);
        free(word);
        start_pos =i+1;

    }
    else if (curr_char == '\"'){        /*If a quote was found, copy the rest of the string and exit loop*/
        word= malloc(len-i+1*(sizeof(char)));
        strncpy(word,command+i,len-i);
        word[len-i+1]='\0';
        printf("%s\n",word);
        free(word);
        quote=1;
        break;
    }

}if (quote==0){                 /*If there was no quote in the string, get the last element*/
    word = malloc(len-start_pos+1*(sizeof (char)));
    strncpy(word,command+start_pos,len-start_pos);
    word[len-start_pos+1]='\0';
    printf("%s\n",word);
        free (word);
    }
    return 0;
}
int main(){
    char buffer[80] = "Hello   \"WOrld  test\"";
    process_command(buffer);
    return 0;
}

但是,我想知道这是否是一种有效的标记方式?这是为了处理用户输入的指令。所以如果用户输入

add 1 2 "SOme text"

我想将查询标记为三个部分,然后对其进行处理。所以要做到这一点,我将它标记化,并将它们推入队列,稍后我可以将项目一个接一个地弹出并处理它。

4

2 回答 2

1

您必须确保为要执行的 strncpy 分配足够的内存。这两行相差一个,因为strncpy也写入了一个零字节:

word = malloc(i-start_pos*(sizeof(char)));
strncpy(word,command+start_pos,i-start_pos);

这两行没有意义:您分配 len-i 字节,然后将 len 字节(加上零字节)写入它:

word = malloc(len-i*(sizeof(char)));
strncpy(word,command+i,len);
于 2012-07-04T00:35:34.630 回答
1

您有几个问题,其中一些是:

for(i=0,start_pos=0;i<strlen(command);i++){
    curr_char = command[i];
    if (curr_char == ' '){
        if (command[i-1]==' ') {start_pos++;continue;}      // accesses an invalid array offset when i == 0
        word = malloc(i-start_pos*(sizeof(char)));          // doesn't allocate space for null terminator
        strncpy(word,command+start_pos,i-start_pos);        // doesn't null terminate the string
        printf("%s\n",word);
        free(word);

        start_pos =i+1;

    }
    else if (curr_char == '\"'){
        word= malloc(len-i*(sizeof(char)));                 // doesn't allocate space for null terminator
        strncpy(word,command+i,len);                        // writes past the end of the allocated buffer
        printf("%s\n",word);
        free(word);
        i=len+len;                                          // not sure what the intent of this is?  use `break;`?
    }

}

一般来说,strncpy()应该避免,因为它并不总是符合用户的期望,因此它经常涉及错误代码。

此外,您的报价处理非常简单 - 它假定引用的项目始终是字符串中的最后一个标记。这可能是您想要的,但它不适用于一组令牌,例如:

"one and two" three
于 2012-07-04T00:50:55.273 回答