Autor
|
Tema: [C] Segmentation Fault - Fallo manejo de cadenas (Leído 3,091 veces)
|
Miky Gonzalez
Desconectado
Mensajes: 87
http://mikygonzalez.comule.com/blog/
|
Voy a ir directo: tengo una función en C que me provoca un fallo de segmentación. He revisado y creo que los punteros están bien colocados y no devería de acceder a recursos de memoria no-accesibles. void eliminar_espacios(char *contenido_linea) { char *cadena_temporal = (char *)malloc(255); while(*contenido_linea) { if((*contenido_linea != 32) && (*contenido_linea != '\t')) { contenido_linea++; } else { *cadena_temporal = *contenido_linea; cadena_temporal++; } } return (void)contenido_linea; }
Elimina los espacios de una cadena. Saludos!
|
|
« Última modificación: 20 Marzo 2013, 20:30 pm por Miky Gonzalez »
|
En línea
|
Mi blog personal, con información acerca de programación, seguridad, desarrollo y electrónica: EN CONSTRUCCIÓN
|
|
|
amchacon
Desconectado
Mensajes: 1.211
|
El fallo está en la condición del while: while(*contenido_linea)
Esa condición siempre va a ser cierta, incluso aunque se sobrepase la cadena (no tienes garantía que haya un NULL al final!). Está es la alternativa que te propongo: # include <stdio.h> # include <stdlib.h> # include <string.h> int main() { char* contenido_linea = "Cosita linda y hermosa"; char* cadena_temporal = (char *)malloc(255); int tamanyo = strlen(contenido_linea ); int i = 0; int posicion = 0; for (i = 0; i < tamanyo;i++) { if((contenido_linea[i] != 32) && (contenido_linea[i] != '\t')) { cadena_temporal[posicion] = contenido_linea[i]; posicion++; } } cadena_temporal[posicion] = '\0'; // Añadimos el caracter nulo al final return 0; }
|
|
« Última modificación: 20 Marzo 2013, 21:14 pm por amchacon »
|
En línea
|
|
|
|
x64core
Desconectado
Mensajes: 1.908
|
El fallo está en la condición del while: while(*contenido_linea)
Esa condición siempre va a ser cierta, incluso aunque se sobrepase la cadena (no tienes garantía que haya un NULL al final!). Está es la alternativa que te propongo: # include <stdio.h> # include <stdlib.h> # include <string.h> int main() { char* contenido_linea = "Cosita linda y hermosa"; char* cadena_temporal = (char *)malloc(255); int tamanyo = strlen(contenido_linea ); int i = 0; int posicion = 0; for (i = 0; i < tamanyo;i++) { if((contenido_linea[i] != 32) && (contenido_linea[i] != '\t')) { cadena_temporal[posicion] = contenido_linea[i]; posicion++; } } cadena_temporal[posicion] = '\0'; // Añadimos el caracter nulo al final return 0; }
Que quieres decir que la condición while es incorrecta? Igual si se le pasa a strlen una secuencia de bytes sin ninguna garantización que haya un null, también fallaria. o algun valor incorrecto, lo mejor seria depurar el codigo. Por cierto, si yo pasara una cadena con una longitud de 1000 con ningun espacio y el buffer reservado es de 255 bytes? mi version: PCHAR StrRemoveSpace(PCHAR lpStr) { PCHAR lpStrRet; PCHAR lpStrIndx; SIZE_T MaxSizeOfStr; if(!lpStr) return FALSE; MaxSizeOfStr = strlen(lpStr); if(!(MaxSizeOfStr > 0)) return FALSE; lpStrRet = lpStrIndx = (PCHAR) malloc(MaxSizeOfStr); if(!lpStrRet) return FALSE; for(INT x = 0; x < MaxSizeOfStr+1; x++) { if(lpStr[x] != 32) *lpStrIndx++ = lpStr[x]; } return lpStrRet; }
Comprueba cualquier tipo de valor incorrecto o no esperado, y para más seguridad podrias agregar un controlador de excepcion, debido a que podria aún darse una.
|
|
« Última modificación: 20 Marzo 2013, 21:47 pm por x64Core »
|
En línea
|
|
|
|
Khronos14
Desconectado
Mensajes: 443
A lie is a lie
|
amchacon, el bucle está perfecto. Lo que hace es comprobar que el carácter actual es distinto de 0, que es el terminador de cadena. Yo creo que el problema puede estar en que para cadena_temporal reservas 255 bytes y si contenido_linea tiene más caracteres = Segmentation Fault. void eliminar_espacios(char *contenido_linea) { char *cadena_temporal = (char *)malloc(strlen(contenido_linea ) + 1); while(*contenido_linea) { if((*contenido_linea != 32) && (*contenido_linea != '\t')) { contenido_linea++; } else { *cadena_temporal = *contenido_linea; *(++cadena_temporal) = '\0'; } } return (void)contenido_linea; }
Saludos.
|
|
|
En línea
|
|
|
|
x64core
Desconectado
Mensajes: 1.908
|
amchacon, el bucle está perfecto. Lo que hace es comprobar que el carácter actual es distinto de 0, que es el terminador de cadena. Yo creo que el problema puede estar en que para cadena_temporal reservas 255 bytes y si contenido_linea tiene más caracteres = Segmentation Fault. void eliminar_espacios(char *contenido_linea) { char *cadena_temporal = (char *)malloc(strlen(contenido_linea ) + 1); while(*contenido_linea) { if((*contenido_linea != 32) && (*contenido_linea != '\t')) { contenido_linea++; } else { *cadena_temporal = *contenido_linea; *(++cadena_temporal) = '\0'; } } return (void)contenido_linea; }
Saludos. ??? - Por cierto, alguien me puede explicar que es esto: void eliminar_espacios(char *contenido_linea) { ... return (void)contenido_linea;
|
|
|
En línea
|
|
|
|
amchacon
Desconectado
Mensajes: 1.211
|
amchacon, el bucle está perfecto. Lo que hace es comprobar que el carácter actual es distinto de 0, que es el terminador de cadena. Yo creo que el problema puede estar en que para cadena_temporal reservas 255 bytes y si contenido_linea tiene más caracteres = Segmentation Fault. void eliminar_espacios(char *contenido_linea) { char *cadena_temporal = (char *)malloc(strlen(contenido_linea ) + 1); while(*contenido_linea) { if((*contenido_linea != 32) && (*contenido_linea != '\t')) { contenido_linea++; } else { *cadena_temporal = *contenido_linea; *(++cadena_temporal) = '\0'; } } return (void)contenido_linea; }
Saludos. Estas añadiendo caracteres nulos en cada iteración? Oo Solo tienes que añadirselo al final. Por cierto, alguien me puede explicar que es esto: void eliminar_espacios(char *contenido_linea) { ... return (void)contenido_linea; Pues es una buena pregunta, al pasar un cast como void nos quedaría algo así: void eliminar_espacios(char *contenido_linea) { ... return; Es decir, no devuelves nada.
|
|
|
En línea
|
|
|
|
x64core
Desconectado
Mensajes: 1.908
|
Pues es una buena pregunta, al pasar un cast como void nos quedaría algo así: void eliminar_espacios(char *contenido_linea) { ... return; Es decir, no devuelves nada. Seguro, yo realmente no entiendo lo que querian ellos hacer. - Por cierto, El codigo de Khronos14 genera una excepción.
|
|
|
En línea
|
|
|
|
durasno
Desconectado
Mensajes: 373
|
Elimina los espacios de una cadena. No, lo unico q hace es copiar los espacios en cadena temporal... El problema de fallo de segmentación se debe a que estas generando un "bucle infinito". Supongamos que tenemos la cadena "Hola mundo", usando tu funcion la condicion del if va a hacer verdadera hasta que contenido_linea apunte al espacio. Una vez que apunte al espacio se ejecuta el else, pero que sucede en el ciclo siguiente y en los posteriores??? contenido_linea va a seguir apuntando al espacio ya que no lo incrementas fuera del if, entonces solo se ejecuta el else y la condicion del while siempre es verdadero. Al entrar solo al else se incrementa permanentemente el puntero cadena_temporal provocando el "fallo de segmentación" Espero se haya entendido. Saludos
|
|
|
En línea
|
Ahorrate una pregunta, lee el man
|
|
|
|
|