我这里有一个稍微修改的JISON计算器示例版本:
/* description: Parses end executes mathematical expressions. */
/* lexical grammar */
%lex
%%
s+ /* skip whitespace */
[0-9]+("."[0-9]+)?b return 'NUMBER'
"*" return '*'
"/" return '/'
"-" return '-'
"+" return '+'
"^" return '^'
"!" return '!'
"%" return '%'
"(" return '('
")" return ')'
"PI" return 'PI'
"E" return 'E'
<<EOF>> return 'EOF'
. return 'INVALID'
/lex
/* operator associations and precedence */
%left '+' '-'
%left '*' '/'
%left '^'
%right '!'
%right '%'
%left UMINUS
%start expressions
%% /* language grammar */
expressions
: e EOF
{ typeof console !== 'undefined' ? console.log($1) : print($1);
return $1; }
;
e
: e '+' e
{$$ = $1+$3;}
| e '-' e
{$$ = $1-$3;}
| e '*' e
{$$ = $1*$3;}
| e '/' e
{$$ = $1/$3;}
| e '^' e
{$$ = Math.pow($1, $3);}
| e '!'
{{
$$ = fact($1);
}}
| e '%'
{$$ = $1/100;}
| '-' e %prec UMINUS
{$$ = -$2;}
| '(' e ')'
{$$ = $2;}
| NUMBER
{$$ = Number(yytext);}
| E
{$$ = Math.E;}
| PI
{$$ = Math.PI;}
;
%%
/*why doesn't this work at runtime?
I see other examples defining declarations this way but I must be doing something wrong
I couldn't find a syntactically valid way of putting this declaration anywhere but here,
which is probably the issue*/
function fact(n) {
var tot=1;
for(var i=2;i<=n;++i) {
tot*=i;
}
return tot;
}
请注意!
运算符定义中的细微差异。 我正在尝试在外部定义fact
函数,而不是内联定义。
截至目前,它在运行时告诉我 fact is not defined
. 我该如何解决这个问题? 另外,为什么计算器示例在阶乘定义周围使用两个大括号,{{ /*like so*/ }}
?
要调用在您的作品下定义的函数(例如 fact
),您可以使用 mod-大括号表示法 %{
和 %}
进行多行语义操作:
e
: e '+' e
...
| e '!'
%{
// the %{ tells jison this is a multi-line js eval statement
$$ = fact($1);
%}
;
作为最终解决方案,请尝试以下操作:
/* lexical grammar */
%lex
%%
s+ /* skip whitespace */
[0-9]+("."[0-9]+)?b return 'NUMBER'
"*" return '*'
"/" return '/'
"-" return '-'
"+" return '+'
"^" return '^'
"!" return '!'
"%" return '%'
"(" return '('
")" return ')'
"PI" return 'PI'
"E" return 'E'
<<EOF>> return 'EOF'
. return 'INVALID'
/lex
/* operator associations and precedence */
%left '+' '-'
%left '*' '/'
%left '^'
%right '!'
%right '%'
%left UMINUS
%start expressions
%% /* language grammar */
expressions
: e EOF
%{
typeof console !== 'undefined' ? console.log($1) : print($1);
return $1;
%}
;
e
: e '+' e
{$$ = $1+$3;}
| e '-' e
{$$ = $1-$3;}
| e '*' e
{$$ = $1*$3;}
| e '/' e
{$$ = $1/$3;}
| e '^' e
{$$ = Math.pow($1, $3);}
| e '!'
%{
$$ = fact($1);
%}
| e '%'
{$$ = $1/100;}
| '-' e %prec UMINUS
{$$ = -$2;}
| '(' e ')'
{$$ = $2;}
| NUMBER
{$$ = Number(yytext);}
| E
{$$ = Math.E;}
| PI
{$$ = Math.PI;}
;
%%
function fact(n) {
var tot=1;
for(var i=2;i<=n;++i) {
tot*=i;
}
return tot;
}
正如@Nucleon和@chaosbohne所指出的,接受的答案是不正确的。
直接来自Jison维基(https://github.com/zaach/jison/wiki/Deviations-From-Flex-Bison)
在词法分析器操作中,如果要使用块样式语句,请使用 %{ ... %} 分隔符,例如:
.* %{
if (true) {
console.log('test');
}
// ...
%}
在解析器操作中,使用 {{ .. }} 分隔符来实现相同的目的。
因此,如果您在词法分析器中,对于多行操作,请使用 %{ action %}
.在分析器中,对于多行操作,请使用 {{ action }}
。如果您正在编写单行操作,则单大括号{}
词法分析和解析器中正常工作。