为什么 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
,这已经发生了并且调用 strtok
时 argv[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);
}
我正在编写此程序以将字符串转换为 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
,这已经发生了并且调用 strtok
时 argv[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);
}