入力文字列を解析するマルチスレッドRPCサーバーがあります。Scalaのパーサーコンビネーターライブラリがマルチスレッドセーフではないという問題が発生しました。Parsers.scalaのvarlastNoSuccessがすべての解析で使用されます。この行でNullPointerExceptionが発生します
if (!(lastNoSuccess != null && next.pos < lastNoSuccess.next.pos))
パーサーの1つを拡張するオブジェクトを作成することによってパーサーを実装するデフォルトの方法ですが、それぞれが独自の内部状態を持つようにオンデマンドでパーサーを構築したいので、オブジェクトの代わりにクラスを使用しています。ただし、結果のパターンマッチが必要なため、コンパイルできません。
import scala.util.parsing.combinator.RegexParsers
class SqlParserImpl
extends RegexParsers
{
val term: Parser[String] = """(?i)term\b""".r
}
object Test
{
def main(args: Array[String]): Unit =
{
val parser = new SqlParserImpl
parser.parseAll(parser.term, "term") match {
// How do I match?
case SqlParserImpl#Success(result, _) => true
case SqlParserImpl#NoSuccess => false
}
}
}
で失敗する
t.scala:16: error: '=>' expected but '#' found.
case SqlParserImpl#Success(result, _) => true
^
t.scala:17: error: '=>' expected but '#' found.
case SqlParserImpl#NoSuccess => false
^
two errors found