- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
想象一下以下类型的字符串:
if ((a1 and b) or (a2 and c)) or (c and d) or (e and f)
现在,我想得到括号中的表达式,所以我用以下语法编写了一个 PEG
解析器:
from parsimonious.grammar import Grammar
grammar = Grammar(
r"""
program = if expr+
expr = term (operator term)*
term = (factor operator factor) / factor
factor = (lpar word operator word rpar) / (lpar expr rpar)
if = "if" ws
and = "and"
or = "or"
operator = ws? (and / or) ws?
word = ~"\w+"
lpar = "("
rpar = ")"
ws = ~"\s*"
""")
解析得很好
tree = grammar.parse(string)
现在问题来了:如何为这棵树编写一个 NodeVisitor
类来只获取因子?我的问题是第二个分支可以深度嵌套。
def walk(node, level = 0):
if node.expr.name == "factor":
print(level * "-", node.text)
for child in node.children:
walk(child, level + 1)
walk(tree)
但实际上无济于事(因素重复出现)。
注意:此问题基于 another one在 StackOverflow 上。
最佳答案
How would I go about it to get ((a1 and b) or (a2 and c)), (c and d) and (e and f) as three parts?
您可以创建一个访问者,当解析树中的节点是 (
,其中增加了深度变量,并且当 )
时“监听”遇到时,深度变量减小。然后在调用的匹配括号表达式的方法中,检查深度,然后将其添加到要从访问者返回的表达式列表中。
这是一个简单的例子:
from parsimonious.grammar import Grammar
from parsimonious.nodes import NodeVisitor
grammar = Grammar(
r"""
program = if expr+
expr = term (operator term)*
term = (lpar expr rpar) / word
if = "if" ws
and = "and"
or = "or"
operator = ws? (and / or) ws?
word = ~"\w+"
lpar = "("
rpar = ")"
ws = ~"\s*"
""")
class ParExprVisitor(NodeVisitor):
def __init__(self):
self.depth = 0
self.par_expr = []
def visit_term(self, node, visited_children):
if self.depth == 0:
self.par_expr.append(node.text)
def visit_lpar(self, node, visited_children):
self.depth += 1
def visit_rpar(self, node, visited_children):
self.depth -= 1
def generic_visit(self, node, visited_children):
return self.par_expr
tree = grammar.parse("if ((a1 and b) or (a2 and c)) or (c and d) or (e and f)")
visitor = ParExprVisitor()
for expr in visitor.visit(tree):
print(expr)
打印:
((a1 and b) or (a2 and c))
(c and d)
(e and f)
关于python - Python 中用于 PEG 解析器的 NodeVisitor 类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55405055/
使用语法 start = b / a a = "4" "2" b = "4" 与 peg.js识别 4 但不识别 42 在这种情况下错误 “第 1 行,第 2 列:预期的输入结束,但找
我试图了解如何使用 PEG.js 在文本中进行简单的搜索/替换。当然,这不是解析器的预期用途,但无论如何我对这些语言背后的逻辑感到好奇,以产生一些搜索/替换。 我遇到的问题是很难积极定义某些定义的互补
扩展我之前的帖子,我还在写汉诺塔。在解释了如何在钉子上画环的绝妙解决方案之后,我仍然有一个问题,我已经摆弄了很长一段时间了。 这是我的 PegClass: namespace Towers_Of_Ha
我最近遇到了 PEG 解析器和 Guido van Rossum 的 article on PEG parsers以及如何构建它们。那篇文章讨论了“PEG”解析器,但在内部它看起来就像一个递归下降解析
我的同事 PaulS 问了我以下问题: 我正在为现有语言(SystemVerilog - IEEE 标准)编写一个解析器,并且该规范中有一条结构与此类似的规则: cover_point =
我写了一个 PEG 解析器生成器只是为了好玩(有时间我会在 NPM 上发布它),并认为在它上面添加一个随机短语生成器会很容易。这个想法是在给定语法的情况下自动获得正确的短语。所以我设置了以下规则来从每
我很喜欢使用 YARD: http://www.ootl.org/yard/ http://code.google.com/p/yardparser/ http://www.codeproject.c
我正在使用 rust-peg 实现类似 OCaml 的语言我的解析器有一个错误。 我定义了 if 语句语法,但它不起作用。 我猜测试用例输入被解析为 Apply(Apply(Apply(Apply(f
我想使用 Peg.js 来解析和验证我一直在使用正则表达式所做的事情,但我正在努力解决语法问题。 我的 Peg.js 程序是: start = (var / other) cr d:var {retu
我正在尝试使用pegjs创建一个解析器。我需要解析类似的内容: blah blah START Lorem ipsum dolor sit amet, consectetur adipiscing e
我目前正在编写PEG.js语法,并且希望它输出自定义错误。例如,我目前具有用于创建函数的这种结构。 //Function Declaration FUNCTION_DECLARATION = FUNC
我当前遇到的问题是我的代码无法解决钉子纸牌的不同变体。我的测试程序测试了 4 个简单的可解板。 (1步解法)上一步、下一步、左一步、右一步。我的代码毫无问题地解决了这些问题,并测试了无法解决的板。我遇
使用 https://github.com/JetBrains/Grammar-Kit如何不用左递归重写文法? grammar ::= exprs exprs::= (sum_expr (';')?)
我正在尝试解析逗号分隔的列表。为了简单起见,我只使用数字。这些表达式是有效的: (1,4,3) () (4) 我可以想到两种方法来做到这一点,我想知道为什么失败的例子不起作用。我相信它是正确的 BNF
我将如何在 Treetop 中做这样的事情? /.+?;/ 似乎唯一的方法是这样做: [^;]+ ';' 哪个有点丑......还有其他方式吗? .+? 似乎不起作用.. 最佳答案 默认情况下 PEG
我在定义允许两个标签(包括那些标签)之间的任何文本的表达式时遇到问题。 例子: #ifdef asdasdasdasdasdsasd asdasdasdasdasdasdasd asdasdasda
我有以下 peg.js 脚本: start = name* name = '** name ' var ws 'var:' vr:var ws 'len:' n:num? ws 'label:' lb
我正在尝试创建一个钉跳拼图,并且正在执行格式说明符,我意识到算法尚未完成,但有人可以解释为什么我的条件被忽略吗?即使列不止一列,它也会运行格式规范。例如,在钉跳益智游戏中,第一行之后的每一行的每个“+
我在 PEG.js 和匹配单词方面遇到问题。 它看起来像这样:Words = "stack"/"overflow"/"stackoverflow" - 当我尝试匹配“stackoverflow”时,它
编辑:您可以在此处关注进度:https://github.com/simple-updates/template 我正在使用 peg.js并尝试编写一些可以解释模板的东西,例如: hello {{ "
我是一名优秀的程序员,十分优秀!