Skip to content

sofiagermer/PFL-PROJECT

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

73 Commits
 
 
 
 
 
 
 
 

Repository files navigation

PFL-PROJECT-G05

BigNumber.hs


1. Definição do tipo BigNumber

Um data type em que os dígitos de um numero são armazenados numa lista (em ordem inversa) e que pode ser positivo, Positive, ou negativo, Negative.

Número positivo :

Positive[3]

Número Negativo :

Negative[3]


Definição de custom Ord instance

Com o intuito de tornar o código mais compreensível e legível, foi feito o overload aos operadores <= (instance Ord BigNumber).

1 BigNumber Negativo e 1 BigNumber Positivo

Retorna sempre True porque um número negativo é sempre menor que um positivo.

Negative [3] < Positive [0,1] = True

1 BigNumber Positivo e 1 BigNumber Negativo

Retorna sempre False porque um número positivo é sempre maior que um positivo.

Positive [3] < Negative [0,1] = False

2 BigNumbers Negativos:

  1. Se tiverem o mesmo comprimento é comparado o valor das listas invertidas .

Negative [0,3] < Negative [0,1] = True

  1. Se o comprimento de BigNumber1 menor que BigNumber2 retorna False

Negative [3] < Negative [0,2] = False

  1. Em todos os outros casos True

2 BigNumbers Positivos

  1. Se tiverem o mesmo comprimento é comparado o valor das listas invertidas.

Positive [0,3] < Positive [0,1] = False

  1. Se o comprimento de BigNumber1 menor que BigNumber2 retorna True

Positive [3] < Positive [0,2] = True

  1. Em todos os outros casos False

2. Scanner

Função que converte uma string em big-number.

scanner "1234" = Positive [4,3,2,1]

scanner "1234" = Negative [4,3,2,1]


3. Output

Função que converte um Big-Number em string.

output (Positive [4,3,2,1]) = "1234"

output (Negative [4,3,2,1]) = "-1234"


4. Soma

Sinais : somaBN

1. Soma de 2 BigNumbers Positivos

Resultado Positivo e é efetuada a soma ao respetivo valor absoluto do BigNumber.

somaBN (Positive[3])(Positive[2]) = Positive [5]

2. Soma de 2 BigNumbers negativos

Resultado Negativo e é efetuada a soma ao respetivo valor absoluto do BigNumber.

somaBN (Negative[3])(Negative[2]) = Negative [5]

3. Soma de 1 BigNumber Positivo com 1 BigNumber Negativo

É efetuada a subtração do BigNumber Positivo pelo BigNumber Negativo, considerando ambos os BigNumbers Positivos.

somaBN (Positive[3])(Negative[2]) = Positive [1]

4. Soma de 1 BigNumber Negativo com 1 BigNumber Negativo

É efetuada a subtração do BigNumber Positivo pelo BigNumber Negativo, considerando ambos os BigNumbers Positivos.

somaBN (Negative[3])(Positive[2]) = Negative [1]

Soma de números com quantidades diferentes de dígitos :somaBNAux

5. Soma de BigNumber1 com BigNumber2 , onde BigNumber1 tem mais dígitos

Enchemos BigNumber2 com zeros não significativos de forma a ambos ficarem com o mesmo número de dígitos.

Soma efetuada normalmente

6. Soma de número BigNumber1 com número BigNumber2 , onde BigNumber2 tem mais dígitos

Enchemos BigNumber1 com zeros não significativos de forma a ambos ficarem com o mesmo número de dígitos.

Soma efetuada normalmente

Soma de números com número igual de dígitos : somaBNAux1

7. Soma normal de duas listas de Dígitos

Algoritmo com Transporte : Começamos a iterar as listas dos dígitos do BigNumber somando par a par (exemplo: 1+5 = 6; 2+3 = 5)

somaBN (Positive[1,3])(Positive[5,2]) = Positive [6,5]

8. Soma de dois Dígitos superior a 10

Nesta função existe um campo overflow que serve para assegurar a soma correta nos casos em que a soma de dois dígitos é superior a 10. Assim, tem o valor default de 0, e assume o valor do "excesso":

No exemplo seguinte a soma 8 + 2 = 10 tem resto 0 e divisão inteira de 1 pelo que na soma dos dígitos seguintes é adicionado 1 a esse valor:

somaBN (Positive[8,2,1])(Positive[2,1,1]) = Posittive [0,4,2]

9. Resultado da soma tem número superior de dígitos a qualquer um dos números

Quando o o "overflow" da soma é diferente de 0, acrescenta esse dígito no campo mais significativo do número (ex: 8 + 2 = 10) -> overflow = 1.

somaBN (Positive[8])(Positive[2]) = Positive [0,1]


5 . Subtração

Sinais

1. Subtração de um BigNumber Positivo por um Negativo

Neste caso o sinal da operação é Positivo e é efetuada a soma dos valores absolutos dos BigNumbers.

subBN (Positive[3])(Negative[2]) = Positive[5]

2. Subtração de um BigNumber Positivo por um Negativo

Neste caso o sinal da operação é Negativo e é efetuada a soma dos valores absolutos dos BigNumbers.

subBN (Negative[3])(Positive[2]) = Negative[5]

3. Subtração entre 2 BigNumbers Positivos

Quando o caso 2.2 não se aplica, procedimento 2.3 é aplicado.

subBN (Positive[3])(Positive[2]) = Positive[1]

4. Soma de 2 BigNumber Negativos

Quando o caso 2.2 não se aplica, procedimento 2.3 é aplicado.

subBN (Negative[3])(Positive[2]) = Negative[5]

Quando o resultado da subtração é 0

1. Subtração de BigNumbers com sinal e valor absoluto iguais

Neste caso, não se segue para a função auxiliar para calcular normalmente a subtração e devolve- se simplesmente Positive [0].

subBN (Positive[3])(Positive[3]) = Positive[0]

subBN (Negative[3])(Negative[3]) = Positive[0]

Quando o valor absoluto dos BigNumbers é diferente

1. Lista de Dígitos de BigNumber1 maior que lista de Dígitos de BigNumber2

Neste caso, tal como na soma, enchemos BigNumber2 com zeros não significativos de forma a ambos ficarem com o mesmo número de dígitos.

subBN (Positive[0,3])(Positive[2]) = Positive[8,2]

2. Lista de Dígitos de BigNumber1 menor que lista de Dígitos de BigNumber2

Neste caso, tal como na soma, enchemos BigNumber1 com zeros não significativos de forma a ambos ficarem com o mesmo número de dígitos.

subBN (Positive[3])(Positive[0,2]) = Negative[7,1]

3. Listas de Dígitos de BigNumber1 e BigNumber2 com mesmo tamanho e valor absoluto de BigNumber1 superior ao de BigNumber2

subBN (Positive[0,3])(Positive[0,2]) = Positive[0,1]

4. Listas de Dígitos de BigNumber1 e BigNumber2 com mesmo tamanho e valor absoluto de BigNumber1 inferior ao de BigNumber2

subBN (Positive[0,2])(Positive[0,3]) = Negative[0,1]


3. Multiplicação

Sinais : mulBN

1. Multiplicação de BigNumbers com o mesmo sinal

Resultado tem sinal Positivo e é efetuada a multiplicação do valor absoluto do BigNumber.

mulBN (Positive[3])(Positive[3]) = Positive[9]

mulBN (Negative[3])(Negative[3]) = Positive[9]

1. Multiplicação de BigNumbers sinais diferentes

Resultado tem sinal Positivo e é efetuada a multiplicação do valor absoluto do BigNumber.

mulBN (Positive[3])(Negative[3]) = Negative[9]

mulBN (Negative[3])(Positive[3]) = Negative[9]

Multiplicação de uma lista de dígitos por um valor : mulAux1

Recorrendo à função map do prelúdio, multiplicamos cada dígito dessa lista por esse valor.

mulAux1 [1,2,3,4] 2 = [2,4,6,8]

Multiplicação das listas de dígitos de BigNumbers : mulAux2

1. Multiplicação de uma lista por outra com tamanho superior a 1

Por cada dígito da segunda lista, multiplicamos todos os dígitos da primeira lista por esse valor.

Variável counter inicializada a 0, sendo incrementada (+1) por cada chamada à função mulAux1.

Depois de se efetuar a multiplicação da primeira lista por cada elemento da segunda, somam se todas as listas obtidas com o auxílio da função somaBN explicada anteriormente.

mulBN (Positive [1,3]) (Positive [2,1]) = Positive [2,7,3]

NOTA :

Neste exemplo multiplicamos [1,3] por [1] = [1,3] e depois [1,3] por [2] = [2,6]

Como é iterado duas vezes [1,3] é enchido por 1 zero ficando [0,1,3] e [2,6] permanece igual

No final somam- se as duas listas [1,3,0] + [2,6] = [2,7,3]


4. Divisão

Sinais

1. Divisão de BigNumbers com o mesmo sinal

divBN (Positive [0,2]) (Positive [4]) = (Positive [5],Positive [0])

divBN (Negative [0,2]) (Negative [4]) = (Positive [5],Positive [0])

2. Divisão de BigNumbers com sinais diferentes

divBN (Positive [0,2]) (Negative [4]) = (Negative [5],Positive [0])

divBN (Negative [0,2]) (Positive [4]) = (Negative [5],Positive [0])

Algoritmo de Divisão

A divisão implementada baseia-se no seguinte algoritmo:

  q=0

  r=n

  while(r>d){
  	r=r-d
  	q=q+1
  }

No final do ciclo obtemos os valores pretendidos, o quociente e o resto da divisão.

Para implementar o algoritmo em Haskell recorremos à recursão, através do uso de uma função auxiliar divBNrecursive que aceita como parâmetros o dividendo, o divisor , o quociente (inicializado a zero pela função divBN) e o sinal do resultado, respectivamente.

Fib.hs

Cálculo do enésimo número de Fibonacci com Integers

Função recursiva : fibRec

Implementação recursiva do cálculo do enésimo número de Fibonacci.

Otimização da função anterior : fibLista

Implementação otimizada da versão recursiva do cálculo do enésimo número de Fibonacci (programação dinâmica).

Recorreu- se a uma lista de resultados parciais tal que (lista !! i) contém o número de Fibonacci de ordem i.

  1. fibLista

  2. fibListaAux1

  3. fibListaAux

Lista infinita com todos os números de Fibonacci e returnar elemento n

Implementação do cálculo do enésimo número de Fibonacci com auxílio de uma lista infinita com todos os números de Fibonnacci.

Retorna elemento de ordem n.

Cálculo do enésimo número de Fibonacci com BigNumbers

Função recursiva : fibRecBN

Implementação recursiva do cálculo do enésimo número de Fibonacci.

Otimização da função anterior : fibListaBN

Implementação otimizada da versão recursiva do cálculo do enésimo número de Fibonacci (programação dinâmica).

Recorreu- se a uma lista de resultados parciais tal que (lista !! i) contém o número de Fibonacci de ordem i.

  1. fibListaBN

São ignorados os números negativos (returnado BigNumber = Positive [0]) e efetuado o cálculo do respetivo número de Fibonacci para os restantes casos.

  1. fibListaAux1BN

Como os próprios "números" no BigNumber são listas, a lista com os números de fibonacci é , portanto, uma lista de listas.

Restante lógia do algoritmo semelhante à feita com Integers (fibListaAux1).

  1. fibListaAuxBN

Lista infinita com todos os números de Fibonnaci e returnar elemento n : fibListaInfinita

Implementação do cálculo do enésimo número de Fibonacci com auxílio de uma lista infinita com todos os números de Fibonnacci, usando zipWith.

Retorna elemento de ordem n.


Resposta à alínea 4

  Compare as resoluções das alíneas 1 e 3 com tipos (Int-> Int), (Integer->Integer) e (BigNumber->BigNumber), comparando a sua aplicação a números grandes e verificando qual o maior número que cada uma aceita como argumento.

Int é um inteiro que possui limitações de tamanho, sendo que, em computadores de 32bit tem um valor máximo de 2147483647 e um valor mínimo de -2147483648.

Integer também é um int, no entanto, não tem limitações de tamanho. Como consequência pode ser usado para representar números grandes, acabando por ser menos eficiente que uma implementação usando Int.

Tal como o Integer o BigNumber não tem limitações de tamanho, tendo também menos eficiência do que o Int.

Alínea 1 : Abordagem Recursiva

Tipo Número Execution Time
(Int -> Int) 10 0.02 secs
(Integer -> Integer) 10 0.01 secs
(BigNumber -> BigNumber) 10 0.01 secs
(Int -> Int) 20 0.05 secs
(Integer -> Integer) 20 0.06 secs
(BigNumber -> BigNumber) 20 0.12 secs
(Int -> Int) 25 0.17 secs
(Integer -> Integer) 25 0.21 secs
(BigNumber -> BigNumber) 25 0.12 secs
(Int -> Int) 30 1.81 secs
(Integer -> Integer) 30 1.83 secs
(BigNumber -> BigNumber) 30 13.06 secs

Alínea 3 : Abordagem Com Auxílio de Lista Infinita

Tipo Número Execution Time
(Integer -> Integer) 10 0.01 secs
(BigNumber -> BigNumber) 10 0.01 secs
(Integer -> Integer) 20 0.01 secs
(BigNumber -> BigNumber) 20 0.01 secs
(Integer -> Integer) 1000 0.01 secs
(BigNumber -> BigNumber) 1000 0.15 secs

NOTA: Para testar o running time correu- se o código no ghci com :set +s

Conclusão

Uma abordagem recorrendo a Integer ou BigNumber será uma opção ideal quando se pretende usar números grandes.

Atendendo aos tempos de execução concluímos que a abordagem com auxílio de lista infinita é extremamente mais eficiente que a recursiva. Aliás, no caso de Int e Integer estagna nos 0.01 secs para valores grandes.

Por outro lado, esta abordagem também prova ser bastante mais eficiente que a recursiva no caso dos BigNumbers. Porém, como seria de esperar, usando BigNumbers a eficiência é , por consequência, reduzida (devido às inúmeras operações realizadas em listas).


Name E-Mail
Sofia Germer [email protected]
Miguel Lopes [email protected]

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published