Resumo: Dado vários valores grandes e o número de amigos que terão que pagar a conta dizer quanto cada um terá que pagar.
Explicação: Se este problema não tivesse os números grandes seria muito fácil era apenas somar e dividir.
Mas usando as implementações para bignumbers do livro Art of The Contest conseguimos fazer isso com a mesma simplicidade apenas adicionando essas funções.
Vamos ao código:
//Autor: Filipe Areias Névola //Ano: 2008 //Licensa: Você pode usar e alterar, mas deve manter o Autor #include <stdio.h> //Biblioteca padrão de entrada e saída #include <stdlib.h> //Biblioteca onde se encontra o qsort #include <string.h> //Biblioteca para manipulação de char* #define MAX 1000000 #define MAXDIGITS 1001 #define PLUS 1 #define MINUS -1 #define abs(a) a>=0?a:-a //Implementação de bignumbers retirada do livro Art of the Contest typedef struct { char digits[MAXDIGITS]; int signbit; int lastdigit; } bignum; void print_bignum(bignum *); void int_to_bignum(int, bignum *); void char_to_bignum(char *, bignum *); void initialize_bignum(bignum *); int max(int, int); void subtract_bignum(bignum *, bignum *, bignum *); void add_bignum(bignum *, bignum *, bignum *); int compare_bignum(bignum *, bignum *); void zero_justify(bignum *); void divide_bignum(bignum *, bignum *, bignum *); void multiply_bignum(bignum *, bignum *, bignum *); void print_bignum(bignum *n) { int i; if(n->signbit == MINUS) printf("- "); for(i=n->lastdigit; i>=0; i--) printf("%c",'0'+ n->digits[i]); } void int_to_bignum(int s, bignum *n) { int i; int t; if (s >= 0) n->signbit = PLUS; else n->signbit = MINUS; for(i=0; i<MAXDIGITS; i++) n->digits[i] = (char) 0; n->lastdigit = -1; t = abs(s); while(t > 0) { n->lastdigit ++; n->digits[ n->lastdigit ] = (t % 10); t = t / 10; } if(s == 0) n->lastdigit = 0; } void char_to_bignum(char * s, bignum *n) { int i; int j = 0; n->lastdigit = -1; for(i=0; i<MAXDIGITS; i++) n->digits[i] = (char) 0; if(s[0] == '-') { n->signbit = MINUS; for(i=strlen(s)-1; i>=1; i--) { n->lastdigit++; n->digits[j++] = s[i]-'0'; } n->lastdigit = strlen(s)-2; } else { n->signbit = PLUS; for(i=strlen(s)-1; i>=0; i--) { n->lastdigit++; n->digits[j++] = s[i]-'0'; } n->lastdigit = strlen(s)-1; } } void initialize_bignum(bignum *n) { int_to_bignum(0,n); } int max(int a, int b) { if(a > b) return(a); else return(b); } void add_bignum(bignum *a, bignum *b, bignum *c) { int carry; int i; initialize_bignum(c); if(a->signbit == b->signbit) c->signbit = a->signbit; else { if(a->signbit == MINUS) { a->signbit = PLUS; subtract_bignum(b,a,c); a->signbit = MINUS; } else { b->signbit = PLUS; subtract_bignum(a,b,c); b->signbit = MINUS; } return; } c->lastdigit = max(a->lastdigit,b->lastdigit)+1; carry = 0; for(i=0; i<=(c->lastdigit); i++) { c->digits[i] = (char) (carry+a->digits[i]+b->digits[i]) % 10; carry = (carry + a->digits[i] + b->digits[i]) / 10; } zero_justify(c); } void subtract_bignum(bignum *a, bignum *b, bignum *c) { int borrow; int v; int i; initialize_bignum(c); if((a->signbit == MINUS) || (b->signbit == MINUS)) { b->signbit = -1 * b->signbit; add_bignum(a,b,c); b->signbit = -1 * b->signbit; return; } if(compare_bignum(a,b) == PLUS) { subtract_bignum(b,a,c); c->signbit = MINUS; return; } c->lastdigit = max(a->lastdigit,b->lastdigit); borrow = 0; for(i=0; i<=(c->lastdigit); i++) { v = (a->digits[i] - borrow - b->digits[i]); if (a->digits[i] > 0) borrow = 0; if (v < 0) { v = v + 10; borrow = 1; } c->digits[i] = (char) v % 10; } zero_justify(c); } int compare_bignum(bignum *a, bignum *b) { int i; if((a->signbit == MINUS) && (b->signbit == PLUS)) return(PLUS); if((a->signbit == PLUS) && (b->signbit == MINUS)) return(MINUS); if(b->lastdigit > a->lastdigit) return (PLUS * a->signbit); if(a->lastdigit > b->lastdigit) return (MINUS * a->signbit); for(i = a->lastdigit; i>=0; i--) { if(a->digits[i] > b->digits[i]) return(MINUS * a->signbit); if(b->digits[i] > a->digits[i]) return(PLUS * a->signbit); } return 0; } void zero_justify(bignum *n) { while((n->lastdigit > 0) && (n->digits[ n->lastdigit ] == 0)) n->lastdigit --; if((n->lastdigit == 0) && (n->digits[0] == 0)) n->signbit = PLUS; } void digit_shift(bignum *n, int d) { int i; if((n->lastdigit == 0) && (n->digits[0] == 0)) return; for(i=n->lastdigit; i>=0; i--) n->digits[i+d] = n->digits[i]; for (i=0; i<d; i++) n->digits[i] = 0; n->lastdigit = n->lastdigit + d; } void multiply_bignum(bignum *a, bignum *b, bignum *c) { bignum row; bignum tmp; int i,j; initialize_bignum(c); row = *a; for(i=0; i<=b->lastdigit; i++) { for(j=1; j<=b->digits[i]; j++) { add_bignum(c,&row,&tmp); *c = tmp; } digit_shift(&row,1); } c->signbit = a->signbit * b->signbit; zero_justify(c); } void divide_bignum(bignum *a, bignum *b, bignum *c) { bignum row; bignum tmp; int asign, bsign; int i,j; initialize_bignum(c); c->signbit = a->signbit * b->signbit; asign = a->signbit; bsign = b->signbit; a->signbit = PLUS; b->signbit = PLUS; initialize_bignum(&row); initialize_bignum(&tmp); c->lastdigit = a->lastdigit; for(i=a->lastdigit; i>=0; i--) { digit_shift(&row,1); row.digits[0] = a->digits[i]; c->digits[i] = 0; while (compare_bignum(&row,b) != PLUS) { c->digits[i] ++; subtract_bignum(&row,b,&tmp); row = tmp; } } zero_justify(c); a->signbit = asign; b->signbit = bsign; } int main(){ bignum n1,n2,n3,zero; int i,n,d,cont=0; char a[MAX], b[MAX]; while (scanf("%d %d",&n,&d)==2&&n) { scanf("%s",&a); //converte string para bignum char_to_bignum(a,&n1); //+ de 1 elemento, logo tem soma para fazer if(n>1){ for(i=1;i<n;++i){ //lê um novo valor scanf("%s",&b); //converte string para bignum char_to_bignum(b,&n2); //soma n1 com n2 e guarda em n3 add_bignum(&n1,&n2,&n3); //passa o resultado para n1 n1=n3;//passa o resultado para n1 } }else{ //1 elemento não precisa de soma n3=n1; } printf("Bill #%d costs ",++cont); //imprime o total print_bignum(&n3); //converte o divisor para bignum int_to_bignum(d,&n2); //divide o total pelo divisor divide_bignum(&n3,&n2,&n1); printf(": each friend should pay "); //imprime quanto cada um irá pagar print_bignum(&n1); printf("\n\n"); } return 0; }
Alguma dúvida? Pergunte através dos comentários!
Krakóvia - SPOJ - 2285 KRAKOVIA
Em Java deu menos de 30 linhas. :D
Menos linhas digitadas pois com certeza por trás do teu código tem muito mais do que essas simples linhas.
E também menos linhas não significa mais eficiente.
E por último esse tipo de problema é um dos únicos que Java é mais útil que C. Mas esse tipo de problema tende a não cair mais em provas pois são muito facéis.
Obrigado pelo comentário e volte sempre amigo! =] haha
também utilizei menos de 30 linhas utilizando c++.
não foi preciso nenhum método. tudo feito no main mesmo.
Filipe, gostei da idéia de inteiros grandes, mas você não acha um pouco exagerado para um problema tão simples?
Obrigado pelo post. =)
@Den, obrigrado. E sobre a sua implementação, fiquei curioso para saber como fez. Eu usi BigNumbers pois foi a única que pensei. Me mande a sua por e-mail filipe.bico@hotmail.com e eu posto ela aqui também! Abraço!
Vcs são bons mesmo...
Fiquei um tempão nesse problema...
Mas também sou iniciante...
Estou no Quarto Período de Engenharia...
Nada como alguém de Computação...
Mas gosto de programar...
Parabéns pra vcs