6

我有一个工作的 Scala 解析器,但解决方案并不像我想要的那样干净。问题是某些产品必须将空格视为令牌的一部分,但“更高级别”的产品应该能够忽略/跳过空格。

如果我使用扩展较低级别解析器的典型 scala 解析器模式,那么将继承 skipWhitespace 设置,并且事情很快就会变得混乱。

我认为我最好不要使用 extends 方法,而是在高级解析器的类中提供一个低级解析器的实例——但我不确定如何使它工作,这样每个实例都会看到只有一个输入字符流。

这是最低级别解析器的一部分 -

class VulgarFractionParser extends RegexParsers  {
  override type Elem = Char

 override val whiteSpace = "".r

然后我像这样扩展

class NumberParser extends VulgarFractionParser with Positional {

但此时 NumberParser 必须像 FractionParser 一样显式处理空格。对于 NumberParser 它仍然非常易于管理 - 但在下一个级别上,我真的希望能够像普通的 regexParser 那样定义使用空格作为分隔符的产品

一个例子是这样的:

IBM 33.33/ 1200.00
or
IBM 33.33/33.50 1200.00

第二个值有时有两个部分,由“/”分隔,有时只有一个部分,斜线后没有任何内容(甚至根本不包含斜线)。

   def bidOrAskPrice = ("$"?) ~> (bidOrAskPrice1 | bidOrAskPrice2 | bidOrAskPrice3) 

   def bidOrAskPrice1 = number ~ ("/".r) ~ number ~ (SPACES) ^^ { 
     case a ~ slash ~ b ~ sp1    => BidOrAsk(a,Some(b))
  }
  def bidOrAskPrice2 = (number ~ "/" ~ (SPACES)) ^^ { case a ~ slash ~ sp => BidOrAsk(a,None) }
   def bidOrAskPrice3 = (number ~ (SPACES?)) ^^ { case a ~ sp => BidOrAsk(a , None)}
4

2 回答 2

3

一种解决方案是覆盖 handleWhiteSpace 函数并在扩展类中使用 var 值激活跳过空格。

您可以在此处查看 RegexParsers 的代码: https ://github.com/scala/scala/blob/v2.9.2/src/library/scala/util/parsing/combinator/RegexParsers.scala

于 2012-05-30T09:02:33.880 回答
2

将第一个解析器转换为令牌解析器(实际上是词法分析器)并让第二个解析器读取它而不是 plain 不是更有意义Char吗?

于 2012-06-01T05:04:18.840 回答