Quarta Lista de Exercícios

Introduzimos o tipo Acao e suas primitivas para evitar introduzir bugs desnecessariamente no interpretador. O trecho de código abaixo cuida das atribuições em um interpretador hipotético de MicroC que não usa Acao, mas cuja função eval manipula a memória diretamente (essa versão de MicroC não tem o stack pointer nem tratadores de exceção em seu estado, apenas a memória, e não usa continuações):

case Atrib(lval, rval) => lval match {
  case Var(n) => mem => {
    val (rv, nmem) = rval.eval(funs)(env)(mem)
    (rv, mem + (env(n) -> rv))
  }
  case Deref(e) => mem => {
    val (l, nmem1) = e.eval(funs)(env)(mem)
    val (rv, nmem2) = rval.eval(funs)(env)(mem)
    (rv, nmem1 + (l, rv))
  }
  case _ => sys.error("lado esquerdo inválido")
}

1. Quais os bugs do código acima? Mostre trechos de código MicroC que evidenciam os bugs.

Bug 1: No caso Var, a memória que está sendo atualizada com o novo valor não é a resultante dos efeitos colaterais de rval. O código abaixo deveria imprimir 5, mas imprime 4:

a = 1;
b = (a = 2) + 1;
print(a + b)

Bug 2: No caso Deref, a memória que está sendo usada na avaliação de rval não é a resultante dos efeitos colaterais de e. O código abaixo mostra o problema, e deveria imprimir 100, mas imprime 1:

a = 1;
*(a = 100) = a;
print(*100)

Bug 3: No caso Deref, a memória que está sendo usada para o resultado final não está levando em conta os efeitos colaterais de rval, mas apenas os de e. O código abaixo mostra o problema, e deveria imprimir 2, mas imprime 1:

a = 1;
*100 = (a = 2);
print(a)

2. Como você poderia consertar o código acima para corrigir os bugs da questão 1?

case Atrib(lval, rval) => lval match {
  case Var(n) => mem => {
    val (rv, nmem) = rval.eval(funs)(env)(mem)
    (rv, nmem + (env(n) -> rv))
  }
  case Deref(e) => mem => {
    val (l, nmem1) = e.eval(funs)(env)(mem)
    val (rv, nmem2) = rval.eval(funs)(env)(nmem1)
    (rv, nmem2 + (l, rv))
  }
  case _ => sys.error("lado esquerdo inválido")
}

3. Qual a ordem de avaliação dos termos do caso Deref na sua resposta à questão 2 (rval depois e, e depois rval)? Como ficaria o código para inverter a ordem de avaliação?

Primeiro e depois rval. Um jeito de inverter a ordem:

  case Deref(e) => mem => {
    val (rv, nmem1) = rval.eval(funs)(env)(mem)
    val (l, nmem2) = e.eval(funs)(env)(nmem1)
    (rv, nmem2 + (l -> rv))
  }

Ainda continuando com o exemplo acima, suponha que queremos reescrever nosso interpretador para usar continuações, mas novamente sem encapsular as continuações e seus efeitos colaterais em um tipo Acao, mas manipulá-las diretamente em eval.

4. Como seria a assinatura de eval? Novamente, assuma que o estado do interpretador é apenas a memória. Uma continuação desse interpretador recebe um valor e retorna uma função que recebe uma memória e retorna um par de outro valor e outra memória.

A função eval agora retorna uma função que recebe uma continuação e retorna uma computação, e a continuação recebe um valor e retorna uma computação:

def eval(funs: Env[Fun1])(env: Env[Valor]): (Valor => Mem => (Valor, Mem)) => (Mem => (Valor, Mem))

5. Reescreva o código acima para usar as continuações, mantendo os mesmos bugs.

case Atrib(lval, rval) => lval match {
  case Var(n) => k => mem =>
    rval.eval(funs)(env)(rv => nmem => (rv, mem + (env(n) -> rv)))(mem) 
  case Deref(e) => k => mem =>
    e.eval(funs)(env)(l => nmem1 =>
      rval.eval(funs)(env)(rv => nmem2 =>
        (rv, nmem1 + (l -> rv)))(mem))(mem)
  case _ => sys.error("lado esquerdo inválido")
}

6. Reescreva a resposta da questão 2 para usar as continuações, mantendo a mesma ordem de avaliação.

É mais difícil cometer os mesmos erros usando as continuações! O código correto é mais simples:

case Atrib(lval, rval) => lval match {
  case Var(n) => k =>
    rval.eval(funs)(env)(rv => nmem => (rv, nmem + (env(n) -> rv))) 
  case Deref(e) => k =>
    e.eval(funs)(env)(l => 
      rval.eval(funs)(env)(rv => nmem2 =>
        (rv, nmem2 + (l -> rv))))
  case _ => sys.error("lado esquerdo inválido")
}

7. Reescreva a resposta da questão 3 para usar as continuações, mantendo a mesma ordem de avaliação.

  case Deref(e) => k =>
    rval.eval(funs)(env)(rv => 
      e.eval(funs)(env)(l => nmem2 =>
        (rv, nmem2 + (l -> rv))))

Para as questões abaixo, use os interpretadores de fun e proto que estão no projeto da Lista 4.

Objetos e funções de primeira classe são dois lados da mesma moeda. Mesmo a recursão aberta dos objetos pode ser simulada com funções de primeira classe. Por exemplo, o código abaixo mostra uma versão “aberta” da função fatorial, em fun:

let fat = fun (f, n)
            if n < 2 then
              1
            else
              n * (f)(f, n-1)
            end
          end
in
  (fat)(fat, 5)
end

8. Escreva uma função doubler que recebe uma função “aberta” como a função acima e retorna outra função aberta que chama a função passada e dobra o resultado. Se substituirmos o corpo do let acima pela expressão abaixo o resultado do programa deve ser 3840.

let df = doubler(fat) in
  (df)(df, 5)
end

Resposta:

fun doubler(super)
  fun (self, x)
    (super)(self, x) * 2
  end
end

9. Escreva fat e doubler como objetos em proto. Dica: confira o slide 6 da aula 23. O resultado de doubler.make(fat).apply(5) também deve ser 3840.

Note o paralelo com a função acima, lembrando que o parâmetro self é implícito:

object doubler(0)
  def make(f)
    object (0) extends f
      def apply(x)
        super.apply(x) * 2
      end
    end
  end
end

object fat(0)
  def apply(n)
    if n < 2 then 1
    else n * self.apply(n - 1) end
  end
end

10. Qual o resultado de (doubler.make(doubler.make(fat))).apply(5)? É o mesmo de (df)(df, 5) se trocarmos a definição de df acima por doubler(doubler(fat))? Se não for o mesmo é porque sua implementação de doubler tem um bug, conserte!

O resultado é 122880, que é 5 * (4 * (3 * (2 * (1 * 2 * 2) * 2 * 2) * 2 * 2) * 2 * 2) * 2 * 2.

Entrega da lista

Use o formulário abaixo para enviar a sua lista. Lembre de enviar apenas o arquivo package.scala que você modificou. O prazo para envio é quarta-feira, dia 02/08/2013.


Última Atualização: 2016-05-18 10:58