funcion de conversion de bases

Iniciado por m@o_614, 19 Marzo 2014, 01:09 AM

0 Miembros y 3 Visitantes están viendo este tema.

m@o_614

Saludos

tengo una funcion llamada obtener numero, a la que le paso una cadena como argumento algo como #456 y la base(hex,octal,binaria o decimal), y me regresa un int con el numero(ya sin el #) el problema es que si le ingreso el numero #65536 me imprime que el numero es 65535 y no entiendo por qué imprime eso, ya lo habia probado para numeros de otras bases y lo hacia correctamente no se por que ahora no, solo puse la funcion que me da problemas por que el codigo es un poco largo

int obtenerNumero(char *operando,int base)
{
   int i,potencia,num_decimal = 0,lon,entero,x = 1;
   lon = strlen(operando);
   for(i = lon-1,potencia = 0;i >= x;i--,potencia++)
   {
       if(esLetraBase16(operando[i]))
          entero = hexadecimal(operando[i]);
       else
          entero = operando[i]-48;
       num_decimal+= pow(base,potencia)*entero;
       printf("%d\n",num_decimal);
   }
   printf("[%d]",num_decimal);
   return num_decimal;
}

int hexadecimal(char caracter)
{
   int decimal;
   switch(caracter)
   {
       case 'A':case 'a':
          decimal = 10;
          break;
       case 'B':case 'b':
          decimal = 11;
          break;
       case 'C':case 'c':
          decimal = 12;
          break;
       case 'D':case 'd':
          decimal = 13;
          break;
       case 'E':case 'e':
          decimal = 14;
          break;
       case 'F':case 'f':
          decimal = 15;
          break;
       default:
          printf("!Error!\n");
   }
   return decimal;
}


me imprime

6
36
536
5536
65535

gracias de antemano

eferion

6
36
536
5536
65536
[65536]

A mi me sale el resultado correcto.

leosansan

#2
Pues a mí me da bien:

Citar
6
36
536
5536
65536
[65536]
Process returned 0 (0x0)   execution ti
Press any key to continue.



Pero en principio lo que he hecho es prescindir de pow y de la librería math y lo he calculado directamente:

Código (cpp) [Seleccionar]
int obtenerNumero(char *operando,int base)
{
   int i,potencia,num_decimal = 0,lon,entero=0,x = 1;
   lon = strlen(operando);
   for(i = lon-1,potencia = 1;operando[i];i--,potencia*=10)
   {
       /*if(esLetraBase16(operando[i]))
          entero = hexadecimal(operando[i]);
       else*/
          entero = operando[i]-'0';
       num_decimal+= potencia*entero;
       printf("%d\n",num_decimal);
   }
   printf("[%d]",num_decimal);
   return num_decimal;
}
int main(){
 obtenerNumero("65536",10);

 return 0;
}


Pero me quedé con el "mosqueo" de lo que te ocurría. Casi no lo pillo, pero sí, lo he pillado.

El problema radica que el uso de pow implica que la variable sea de tipo float o double y si se usa un int pasan cosas raras, cosa que no sucede con mi método. Es decir, la solución a tu código:


Código (cpp) [Seleccionar]
int obtenerNumero(char *operando,int base)
{
   int i,potencia,lon,entero=0,x = 1;
   float num_decimal = 0; /*<== AQUI ::::::*/
   lon = strlen(operando);
   for(i = lon-1,potencia = 0;operando[i];i--,potencia++)
   {
       /*if(esLetraBase16(operando[i]))
          entero = hexadecimal(operando[i]);
       else*/
          entero = operando[i]-'0';
       num_decimal+= pow(base,potencia)*entero;
       printf("%g\n",num_decimal);
   }
    printf("[%d]",num_decimal);
   return num_decimal;
}
int main(){
 obtenerNumero("65536",10);

 return 0;
}



¡¡¡¡ Saluditos! ..... !!!!



amchacon

#3
El problema radica en el printf("%g"). No se debe usar el %g para enteros.

Podemos comprobar que retorno es correcto
Código (cpp) [Seleccionar]
#include <stdio.h>

int obtenerNumero(char *operando,int base)
{
  int i,potencia,lon,entero=0,x = 1;
  int num_decimal = 0;
  lon = strlen(operando);
  for(i = lon-1,potencia = 0;operando[i];i--,potencia++)
  {
      /*if(esLetraBase16(operando[i]))
          entero = hexadecimal(operando[i]);
       else*/
         entero = operando[i]-'0';
      num_decimal+= pow(base,potencia)*entero;

      //printf("%g\n",num_decimal);
  }
  //printf("[%d]",num_decimal);
  return num_decimal;
}
int main(){
printf("%d\n",obtenerNumero("65536",10));

return 0;
}


Aunque por eficiencia, yo lo haría sin la función pow (tal y como ha hecho Leo).
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar

eferion

Yo le pondría un parche al cálculo de la potencia propuesto por leosansan.


int obtenerNumero(char *operando,int base)
{
    int i,potencia,num_decimal = 0,lon,entero=0,x = 1;
    lon = strlen(operando);
    for(i = lon-1,potencia = 1;operando[i];i--,potencia*=base) // <--- AQUI
    {
        if(esLetraBase16(operando[i]))
           entero = hexadecimal(operando[i]);
        else
           entero = operando[i]-'0';
        num_decimal+= potencia*entero;
        printf("%d\n",num_decimal);
    }
    printf("[%d]",num_decimal);
    return num_decimal;
}


Básicamente para que funcione en bases que no sea la decimal.

leosansan

Cita de: amchacon en 19 Marzo 2014, 12:51 PM
El problema radica en el printf("%g"). No se debe usar el %g para enteros.
.......................................


Creo que no, el problema radica en lo que indiqué.

Una salida con int y %d en el printf:


Citar
6
36
536
5536
65535
[65535]


Y si los declara como float o double e imprime con %d ya ni te cuento.

Cita de: eferion en 19 Marzo 2014, 13:10 PM
Yo le pondría un parche al cálculo de la potencia propuesto por leosansan.

.........................................................
Básicamente para que funcione en bases que no sea la decimal.


Bien observado amigo eferion. Pero ya lo dice la variable: "num_decimal". No creo que ese  fuera el propósito de m@o_614, pero efectivamente el método es lo bastante versátil como para ser usado para distintas bases. Y repito sí tienes razón, su uso es más potente. .....y me has sacado un as que tenía en la manga para otros temas  ;)



¡¡¡¡ Saluditos! ..... !!!!



eferion

Cita de: leosansan en 19 Marzo 2014, 16:37 PM
.....y me has sacado un as que tenía en la manga para otros temas  ;)

jajajaja.

Vaya por dios, siento haberte fastidiado el plan... la próxima avísame por MP para no meter la pata ;)

amchacon

Cita de: leosansan en 19 Marzo 2014, 16:37 PM
Creo que no, el problema radica en lo que indiqué.

Una salida con int y %d en el printf:


Y si los declara como float o double e imprime con %d ya ni te cuento.
¿Me has puesto la salida bien? ¿? :huh:
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar

leosansan

#8
La salida a tu código en mi ordenadorr y compilador:

Citar
6
36
536
5536
65535
EN FUNCION: [65535]
6
36
536
5536
65535
EN FUNCION: [65535]
EN MAIN: 65535

Process returned 0 (0x0)   execution time : 0.028 s
Press any key to continue.



Cita de: amchacon en 19 Marzo 2014, 17:32 PM
¿Me has puesto la salida bien? ¿? :huh:

Te tengo en una muy alta estima, de corazón, como para jugar con esos detalles.

En realidad creo que todo acaba dependiendo del compilador. Fíjate que  a eferion el código de m@o_614 le dio el resultado correcto y sin embargo a  m@o_614  y a mi nos daba mal. ¿¿¿¿¿¿?????..... No encuentro otra explicación que todo acaba dependiendo del compilador y versión usado.

Un fuerte abrazo y todos mis respetos hacia ti.


¡¡¡¡ Saluditos! ..... !!!!



amchacon

Cita de: leosansan en 19 Marzo 2014, 17:53 PMTe tengo en una muy alta estima[/b], de corazón, como para jugar con esos detalles.
Me has entendido mal Leo, no me referia que lo hicieses de mala fe ^^

Lo que quería decir esque me comentas que la salida es erronea y me la enseñas:

Citar6
36
536
5536
65535
EN FUNCION: [65535]
EN MAIN: 65535

Es exactamente la salida que pones en tu primer post. No entiendo, ¿acaso no era esa la salida correcta? :huh:
Por favor, no me manden MP con dudas. Usen el foro, gracias.

¡Visita mi programa estrella!

Rar File Missing: Esteganografía en un Rar