Można użyć "Parser. ^?" operator, aby sprawdzić grupę elementów parsowanych dla duplikatów.
def tokens = tokenA | tokenB | tokenC
def uniqueTokens = (tokens*) ^? (
{ case t if (t == t.removeDuplicates) => t },
{ "duplicate tokens found: " + _ })
Oto przykład, który pozwala wprowadzić dowolny z czterech Stooges w dowolnej kolejności, ale nie do analizowania, czy duplikat napotkaniu:
package blevins.example
import scala.util.parsing.combinator._
case class Stooge(name: String)
object StoogesParser extends RegexParsers {
def moe = "Moe".r
def larry = "Larry".r
def curly = "Curly".r
def shemp = "Shemp".r
def stooge = (moe | larry | curly | shemp) ^^ { case s => Stooge(s) }
def certifiedStooge = stooge | """\w+""".r ^? (
{ case s: Stooge => s },
{ "not a stooge: " + _ })
def stooges = (certifiedStooge*) ^? (
{ case x if (x == x.removeDuplicates) => x.toSet },
{ "duplicate stooge in: " + _ })
def parse(s: String): String = {
parseAll(stooges, new scala.util.parsing.input.CharSequenceReader(s)) match {
case Success(r,_) => r.mkString(" ")
case Failure(r,_) => "failure: " + r
case Error(r,_) => "error: " + r
}
}
}
A niektóre z przykładów użycia:
package blevins.example
object App extends Application {
def printParse(s: String): Unit = println(StoogesParser.parse(s))
printParse("Moe Shemp Larry")
printParse("Moe Shemp Shemp")
printParse("Curly Beyonce")
/* Output:
Stooge(Moe) Stooge(Shemp) Stooge(Larry)
failure: duplicate stooge in: List(Stooge(Moe), Stooge(Shemp), Stooge(Shemp))
failure: not a stooge: Beyonce
*/
}
W tym przypadku każdy token jest właściwością obiektu w stylu json. Polecenie może więc wyglądać tak: "wiadomość todo: link Todo class to database", ponieważ: next wtorek. "Tak więc ogólna reguła zdefiniowana w scala stylu jest podobna do" token = alphanum ~ ": '~ repsep (alphanum,' '). Ale muszę inaczej obsługiwać określone właściwości. –
Musisz się upewnić, że to samo nie występuje więcej niż raz? – ziggystar
Tak, to jest plan, niektóre właściwości są opcjonalne i powinny wystąpić tylko raz. –