为什么 String Array 只能得到 return 第一个元素,而不能得到后续元素?

Why String Array only get to return first element, but not the subsequent element?

我正在编写此程序以将字符串转换为 C 中的字符串数组(使用 char**)。但似乎程序只存储了第一个值的元素,而其余的只是打印(null)。我可以知道我在这里做错了什么吗?

这是我如何测试代码的片段。我正在为 char** 分配内存,并将其传递给 toStrArray 函数以将 argv[1] 中的每个字符串存储到 arg_1(由定界符“”分隔)。 这意味着每次程序看到一个“”,它就标记停止并将字符串存储到数组元素

预期输出:这个城市很漂亮!但是实际输出: This (null) (null) (null)

int main( int argc, char *argv[] )
{
    int i; 
    int numWords; /* Pointer size (row) for arg_1 */
    char **arg_1; /* String array for argument 1 */


    if( argc == ARGUMENT_SIZE ) 
    {
        numWords = getNumWords(argv[1]); /* Get the number of words in arg_1 
                                          eg: "This city is beautiful" will return 4 */

        /* Allocation of string array of arg_1 */
        arg_1 = (char**)malloc(numWords * sizeof(char*));

        /* ASSERTION: Iterate through the rows of string array and allocate their memory */
        for( i = 0; i < numWords; i++ )
            *arg_1 = (char*)malloc(STR_LENGTH * sizeof(char));

        arg_1 = toStrArray( argv[1], arg_1, " " ); /* Converts each word in argv[1] to separated string array 
                                                  so that we can find matching with argv[2] and change it */
    
        for( i = 0; i < numWords; i++ )
            printf("%s ", arg_1[i]);
    }
    return 0;
}

这是我的 toStrArray

char** toStrArray(char str[], char **strArr, char delim[]) 
{
    char *token;
    int i;

    i = 0;
    token = strtok(str, delim);
    /* ASSERTION: Token the string until the end of the string */
    while( token != NULL )
    {
        strArr[i] = token; /* Assigning each word into element array */
        token = strtok(NULL, delim); /* Get the next token (word) */
        i++;
    }
    return strArr;
}

getNumWords 函数:

int getNumWords(char str[])
{
    int num_words;
    char *token;

    token = strtok(str, " ");
    num_words = 0;

    /* ASSERTION: Iterate until we reach the last token */
    while( token != NULL )
    {
        token = strtok(NULL, " ");
        num_words++;
    }
    return num_words;    
}

不能对内存中的同一个字符数组使用 strtok 两次。输入

char str = {'T', 'h', 'i', 's', ' ', 'c', 'i', 't', 'y', ' ', 'i', 's',
            ' ', 'b', 'e', 'a', 'u', 't', 'i', 'f', 'u', 'l', '!',  0};

会变成

char str = {'T', 'h', 'i', 's',   0, 'c', 'i', 't', 'y',   0, 'i', 's',
              0, 'b', 'e', 'a', 'u', 't', 'i', 'f', 'u', 'l', '!',  0};

执行后

char* ret1 = strtok(str,  " "); // Sets the null byte after 'This'
char* ret2 = strtok(NULL, " "); // Sets the null byte after 'city'
char* ret3 = strtok(NULL, " "); // Sets the null byte after 'is'
char* ret4 = strtok(NULL, " "); // Doesn't modify str

请注意,每个空白字符都替换为终止空字节。这样做会将输入字符串切割成多个子字符串:

printf("%s\n", ret1); // Outputs 'This'
printf("%s\n", ret2); // Outputs 'city'
printf("%s\n", ret3); // Outputs 'is'
printf("%s\n", ret4); // Outputs 'beautiful!'

如果你调用 toStrArray,这已经发生了并且调用 strtokargv[1] returns NULL 因为子串 'This' 没有不包含分隔符。

因此我建议用 strchr 来计算分隔符的数量:它 return 指向字符首次出现的指针。从那里继续搜索,直到找不到更多的空格,即 NULL 是 returned.

#include <stdio.h>
#include <string.h>

int main() {
    char str[] = "This city is beautiful!";

    int count = 0;
    for(char* tmp = str; tmp; count++) {
        while (tmp[1] == ' ')     // Ignore preceding and multiple whitespaces
            tmp++;
        tmp = strchr(tmp+1, ' '); // Find next whitespace
    }

    printf("Number of words: %d\n", count);
}