Jump to content

Recommended Posts

Posted

Bom dia/ Boa tarde/ Boa Noite, preciso de uma ajuda com um código. Sou iniciante e não consigo entender um erro no código, estou aprendendo por umas vídeo aulas no youtube e agora estou aprendendo mais sobre funções (mais especificamente funções recursivas). O código já está funcional, mas o meu problema é que não consigo entender um erro (consegui corrigir, mas não entendi o pq dele estar aparecendo), parece que o if afeta o else e consequentemente o resultado final, o código vai estar logo abaixo:

#include <stdio.h>

int main(void) {
  int fatorial(int x);
  int numero, resultado_final;

  printf("Digite um número:\n");
  scanf("%i", &numero);

  resultado_final = fatorial(numero);

  printf("O fatorial de %i é igual a %i\n", numero, resultado_final);
  return 0;
}

int fatorial(int x){
  int resultado;

  if(x < 0){ /* Nesse caso o resultado sempre vai ser zero, mas se mudar o (x < 0) para (x == 0) o else funciona normalmente, ou seja, o if de alguma forma afeta o else que modifica o resultado final (lembrando que isso é aos meus olhos de iniciante), mas quero saber o pq isso acontece??? Podem me ajudar? Por favor. */
    resultado = 1;
  }
  else{
    resultado = x * fatorial(x - 1);
  }

  return resultado;
}

Minha pergunta é: pq o resultado sempre é zero? Se a condição de if não for falsa, não deveria pular para else e executar normalmente??

  • Agradecer 1
Posted

Você leu que estou começando a aprender sobre funções recursivas, né? Mas, meu problema não é com a matemática e sim com a lógica. Pelo que entendi da sua resposta e o pouco que sei sobre funções recursivas, outras funções são criadas na memória até o calculo chegar a zero e depois ela retorna o resultado, mas como o X só pode ser menor que zero, ela não chega até o final e acaba retornando zero, seria isso o que quis dizer? Poderia me indicar algum post/artigo que possa me ajudar? Por favor. Desde já agradeço.

  • Agradecer 1
Posted (edited)

Considere:
 

int fact( int x )
{
  if ( x < 0 ) return 1;

  return x * fact( x - 1 );
}

Se x = 0, inicialmente, temos um return 0 * fact( -1 );
E essa chamada, recursiva, retornará 1 (já que o novo x < 0)... Dai, 0 * 1 = 0.

Note que, por definição, n! = 1 se n <= 1. E fatorial só é definido para valores positivos... A rotina DEVERIA ser:
 

unsigned int fact( unsigned int x )
{
  if ( x <= 1 ) return 1;

  return x * fact( x - 1 );
}

Sendo que a função causará overflow se x > 12.

Edited by fredericopissarra
  • Agradecer 1
Posted

Muito Obrigado, Frederico! Consegui entender, você e os "printf" que espalhei por todo o código me fizeram entender. Nem sabia da existência do overflow, mas depois de dar uma pesquisada entendi o que significa. Como já dizia um grande sábio "Ah Agora eu entendi, agora eu saquei, agora tudo faz sentido, agora todas as peças se encaixaram". De verdade, muito obrigado pela ajuda!

  • Agradecer 1
  • Curtir 1
Posted

Não faz parte de sua dúvida, mas eis um código que não é recursivo e usa aritmética de múltipla precisão e te permite calcular o fatorial de qualquer valor inteiro, positivo, dentro da faixa de um unsigned long int.
 

// Calcula fatorial usando aritmética de multipla precisão.
//
// Compilar com:
//    cc -O2 -o factmp factmp.c -lgmp
//
// Não esquecer de instalar libgmp-dev.
//
#include <stdio.h>
#include <stdlib.h>
#include <gmp.h>
#include <errno.h>
#include <time.h>

int main(int argc, char *argv[])
{
  char *p;
  mpz_t r;
  unsigned long n, c, total;
  time_t t1, t2;

  if ( argc != 2 )
  {
    fprintf( stderr, "Usage: %s <value>\n", argv[0] );
    return EXIT_FAILURE;
  }

  // O valor deve ser inteiro e dentro da faixa de unsigned long int.
  errno = 0;
  n = strtoul( *++argv, &p, 10 );
  if ( errno == ERANGE || *p )
  {
    fputs( "ERROR: Invalid value.\n", stderr );
    return EXIT_FAILURE;
  }

  // Inicializa r e seta-o com 1.
  mpz_init_set_ui(r, 1);

  t1 = time( NULL );

  c = 0;
  total = n;
  while ( n >= 1 )
  {
    mpz_mul_ui( r, r, n-- );
    printf( "%.2f%% complete... \r", 100.0 * ++c / total );
  }
  putchar( '\n' );

  t2 = time( NULL );

  mpz_out_str( stdout, 10, r );
  putchar('\n');

  // Joga r no lixo.
  mpz_clear(r);

  printf( "Total time: %lu seconds.\n", t2 - t1 );

  return EXIT_SUCCESS;
}

Mas, atenção: Se você usar valores muito grandes (maiores que 100000, por exemplo), o tempo de calculo será bem grande (tente!).

[]s
Fred

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.

Guest
Reply to this topic...

×   Pasted as rich text.   Paste as plain text instead

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

  • Recently Browsing   0 members

    • No registered users viewing this page.
×
×
  • Create New...