Problema nested de la función strtok en C

Tengo una cadena como esta:

a;b;c;d;e f;g;h;i;j 1;2;3;4;5 

y quiero analizarlo elemento por elemento. Utilicé la función strtok anidada pero simplemente divide la primera línea y anula el puntero del token. ¿Cómo puedo superar esto? Aquí está el código:

 token = strtok(str, "\n"); while(token != NULL && *token != EOF) { char a[128], b[128]; strcpy(a,token); strcpy(b,a); printf("a:%s\n",a); char *token2 = strtok(a,";"); while(token2 != NULL) { printf("token2 %s\n",token2); token2 = strtok(NULL,";"); } strcpy(token,b); token = strtok(NULL, "\n"); if(token == NULL) { printf("its null"); } } 

Salida:

 token 2 a token 2 b token 2 c token 2 d token 2 e 

No puedes hacer eso con strtok() ; use strtok_r() de POSIX o strtok_s() de Microsoft si están disponibles, o strtok_s() pensar su diseño.

 char *strtok_r(char *restrict s, const char *restrict sep, char **restrict lasts); char *strtok_s(char *strToken, const char *strDelimit, char **context); 

Estas dos funciones son intercambiables. Aunque strtok_s() es una parte opcional de C11 (Anexo K en ISO / IEC 9899: 2011), pocos proveedores distintos de Microsoft han implementado las interfaces en esa sección de la norma.

Con strtok_r ()

 #include  #include  int main(void) { char str[] = "a;b;c;d;e\nf;g;h;i;j\n1;2;3;4;5\n"; char *end_str; char *token = strtok_r(str, "\n", &end_str); while (token != NULL) { char *end_token; printf("a = %s\n", token); char *token2 = strtok_r(token, ";", &end_token); while (token2 != NULL) { printf("b = %s\n", token2); token2 = strtok_r(NULL, ";", &end_token); } token = strtok_r(NULL, "\n", &end_str); } return 0; } 

Resultados

 a = a;b;c;d;e b = a b = b b = c b = d b = e a = f;g;h;i;j b = f b = g b = h b = i b = j a = 1;2;3;4;5 b = 1 b = 2 b = 3 b = 4 b = 5 

Sin strtok_r ()

Esto funciona en contexto, siempre que los datos finalicen con una nueva línea.

 #include  #include  int main(void) { char data[] = "a;b;c;d;e\nf;g;h;i;j\n1;2;3;4;5\n"; char *string = data; char *token = strchr(string, '\n'); while (token != NULL) { /* String to scan is in string..token */ *token++ = '\0'; printf("a = %s\n", string); char *token2 = strtok(string, ";"); while (token2 != NULL) { printf("b = %s\n", token2); token2 = strtok(NULL, ";"); } string = token; token = strchr(string, '\n'); } return 0; } 

Salida

 a = a;b;c;d;e b = a b = b b = c b = d b = e a = f;g;h;i;j b = f b = g b = h b = i b = j a = 1;2;3;4;5 b = 1 b = 2 b = 3 b = 4 b = 5 

strtok_r es la mejor y más segura solución, pero también hay una manera de hacerlo con strtok :

 #include  #include  int main () { char str[] = "a;b;c;d;e\nf;g;h;i;j\n1;2;3;4;5\n"; char *line; char *token; char buf[256]; for (line = strtok (str, "\n"); line != NULL; line = strtok (line + strlen (line) + 1, "\n")) { strncpy (buf, line, sizeof (buf)); printf ("Line: %s\n", buf); for (token = strtok (buf, ";"); token != NULL; token = strtok (token + strlen (token) + 1, ";")) { printf ("\tToken: %s\n", token); } } return 0; } 

Salida:

 Line: a;b;c;d;e Token: a Token: b Token: c Token: d Token: e Line: f;g;h;i;j Token: f Token: g Token: h Token: i Token: j Line: 1;2;3;4;5 Token: 1 Token: 2 Token: 3 Token: 4 Token: 5