progs/display/enfa.scala
author Christian Urban <urbanc@in.tum.de>
Sun, 07 May 2017 03:01:29 +0100
changeset 489 4430477595ec
parent 488 057b4603b940
child 491 7a0182c66403
permissions -rw-r--r--
updated
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
485
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     1
// epsilon NFAs...immediately translated into NFAs
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     2
// (needs nfa.scala)
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     3
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     4
// fixpoint construction
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     5
import scala.annotation.tailrec
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     6
@tailrec
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     7
def fixpT[A](f: A => A, x: A): A = {
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     8
  val fx = f(x)
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
     9
  if (fx == x) x else fixpT(f, fx) 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    10
}
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    11
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    12
// translates eNFAs directly into NFAs 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    13
def eNFA[A, C](starts: Set[A],                     // starting states
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    14
               delta: (A, Option[C]) :=> Set[A],   // epsilon-transitions
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    15
               fins: A => Boolean) : NFA[A, C] = { // final states 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    16
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    17
  // epsilon transitions
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    18
  def enext(q: A) : Set[A] = 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    19
    applyOrElse(delta, (q, None))
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    20
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    21
  def enexts(qs: Set[A]) : Set[A] = 
488
057b4603b940 updated
Christian Urban <urbanc@in.tum.de>
parents: 487
diff changeset
    22
    qs | qs.flatMap(enext(_))     // | is the set-union in Scala
485
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    23
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    24
  // epsilon closure
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    25
  def ecl(qs: Set[A]) : Set[A] = 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    26
    fixpT(enexts, qs)
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    27
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    28
  // "normal" transitions
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    29
  def next(q: A, c: C) : Set[A] = 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    30
    applyOrElse(delta, (q, Some(c)))
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    31
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    32
  def nexts(qs: Set[A], c: C) : Set[A] = 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    33
    ecl(ecl(qs).flatMap(next(_, c)))
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    34
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    35
  // result NFA
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    36
  NFA(ecl(starts), 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    37
      { case (q, c) => nexts(Set(q), c) }, 
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    38
      q => ecl(Set(q)) exists fins)
21dec9df46ba updated
Christian Urban <urbanc@in.tum.de>
parents:
diff changeset
    39
}