- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
出于兴趣,我想学习如何为一种简单的语言编写解析器,并最终为我自己的代码打高尔夫球语言编写解释器,一旦我了解了这些东西的一般工作原理。
所以我开始阅读 Douglas Crockfords 的文章 Top Down Operator Precedence .
注意:如果您想更深入地了解下面代码片段的上下文,您可能应该阅读这篇文章
我很难理解 var
语句和赋值运算符 =
应该一起工作。
华盛顿特区定义一个赋值运算符,如
var assignment = function (id) {
return infixr(id, 10, function (left) {
if (left.id !== "." && left.id !== "[" &&
left.arity !== "name") {
left.error("Bad lvalue.");
}
this.first = left;
this.second = expression(9);
this.assignment = true;
this.arity = "binary";
return this;
});
};
assignment("=");
注意:[[value]] 指的是一个 token ,简化为它的值
现在如果表达式函数达到例如[[t]],[[=]],[[2]]
, [[=]].led
的结果是这样的。
{
"arity": "binary",
"value": "=",
"assignment": true, //<-
"first": {
"arity": "name",
"value": "t"
},
"second": {
"arity": "literal",
"value": "2"
}
}
华盛顿特区使 assignment
功能是因为
we want it to do two extra bits of business: examine the left operand to make sure that it is a proper lvalue, and set an assignment member so that we can later quickly identify assignment statements.
在他介绍 var
语句,定义如下。
The var statement defines one or more variables in the current block. Each name can optionally be followed by = and an initializing expression.
stmt("var", function () {
var a = [], n, t;
while (true) {
n = token;
if (n.arity !== "name") {
n.error("Expected a new variable name.");
}
scope.define(n);
advance();
if (token.id === "=") {
t = token;
advance("=");
t.first = n;
t.second = expression(0);
t.arity = "binary";
a.push(t);
}
if (token.id !== ",") {
break;
}
advance(",");
}
advance(";");
return a.length === 0 ? null : a.length === 1 ? a[0] : a;
});
现在,如果解析器遇到一组标记,如 [[var]],[[t]],[[=]],[[1]]
生成的树看起来像这样。
{
"arity": "binary",
"value": "=",
"first": {
"arity": "name",
"value": "t"
},
"second": {
"arity": "literal",
"value": "1"
}
}
我的问题的关键部分是 if (token.id === "=") {...}
部分。
我不明白为什么我们打电话
t = token;
advance("=");
t.first = n;
t.second = expression(0);
t.arity = "binary";
a.push(t);
而不是
t = token;
advance("=");
t.led (n);
a.push(t);
在...
部分。
这会调用我们的 [[=]]
运营商led
函数(赋值函数),它做
make sure that it is a proper lvalue, and set an assignment member so that we can later quickly identify assignment statements. e.g
{
"arity": "binary",
"value": "=",
"assignment": true,
"first": {
"arity": "name",
"value": "t"
},
"second": {
"arity": "literal",
"value": "1"
}
}
因为没有带有 lbp
的运算符在 0 到 10 之间,调用 expression(0) vs. expression (9)
没有区别。 ( !(0<0) && !(9<0) && 0<10 && 9<10)
)
还有 token.id === "="
条件阻止分配给对象成员为 token.id
要么是 '['
或 '.'
和 t.led
不会被调用。
简而言之,我的问题是:
为什么我们不在变量声明之后可选地调用可用的赋值运算符 led
功能。而是手动设置 first
和 second
声明的成员而不是 assignment
成员(member)?
这里有两个 fiddle 解析一个简单的字符串。使用 original代码和一个使用 assignment运营商led
.
最佳答案
在解析语言时,有两件事很重要 - 语义和句法。
在语义上,var x=5;
和 var x;x=5
即使不完全相同也看起来非常接近(因为在这两种情况下首先声明了一个变量,然后将一个值分配给该声明的变量。这就是您所观察到的并且在大多数情况下是正确的。
在句法上然而,两者不同(这一点很明显)。
在自然语言中,类似的是:
现在要简洁!让我们看看这两个例子。
虽然这两个(几乎)意思相同,但它们显然不是同一句话。回到 JavaScript!
第一个:var x=5
是read the following way :
var x = 5
-----------------------VariableStatement--------------------
var ------------------- VariableDeclarationList
var ------------------- VariableDeclaration
var Identifier ------- Initialiser(opt)
var ------------------- x = AssignmentExpression
var ------------------- x ------------ = LogicalORExpression
var ------------------- x ------------ = LogicalANDExpression
var ------------------- x ------------ = BitwiseORExpression
var ------------------- x ------------ = BitwiseXORExpression
var ------------------- x ------------ = BitwiseANDExpression
var ------------------- x ------------ = EqualityExpression
var ------------------- x ------------ = ShiftExpression
var ------------------- x ------------ = AdditiveExpression
var ------------------- x ------------ = MultiplicativeExpression
var ------------------- x ------------ = UnaryExpression
var ------------------- x ------------ = PostfixExpression
var ------------------- x ------------ = NewExpression
var ------------------- x ------------ = MemberExpression
var ------------------- x ------------ = PrimaryExpression
var ------------------- x ------------ = Literal
var ------------------- x ------------ = NumericLiteral
var ------------------- x ------------ = DecimalLiteral
var ------------------- x ------------ = DecimalDigit
var ------------------- x ------------ = 5
呸!所有这些都必须在语法上发生以解析 var x = 5
,当然,其中很多是处理表达式 - 但它就是这样,让我们检查另一个版本。
这分为两个语句。 变量x; x = 5
第一个是:
var x
--------VariableStatement---
var ---- VariableDeclarationList
var ---- VariableDeclaration
var Idenfifier (optional initializer not present)
var x
第二部分是x=5
,这是一个赋值语句。我可以继续使用同样的疯狂表情 - 但它几乎是一样的。
总而言之,虽然两者在语义上产生相同的结果,但在句法上正如官方语言语法所指定的那样 - 它们是不同的。在这种情况下,结果确实是一样的。
关于javascript - Crockfords 自上而下运算符优先级,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18828321/
int x = 1; System.out.println( x++ + x++ * --x ); 上面的代码打印出“5”,但我不明白怎么办?我一直为最后一个 x 取零,然后乘以仍然为 0 的第二个
我现在正在尝试使用 Preference 类 首选项 pfrOfThis = Preferences.userNodeForPackage(this) 出现错误: “类 java.util.prefs
用下面的代码 import sys print "Hello " + sys.argv[1] if len(sys.argv) > 1 else "Joe" + "." 当我运行时 python he
我的网页包含: td { padding-left:10px; } 引用的样式表包含: .rightColumn * {margin: 0; padding: 0;} 我在 rightc
使用 JPA 我有一个关于 CascadeTypes 的问题。 例如: @ManyToMany(fetch=FetchType.LAZY, cascade={CascadeType.PERSIST,
下面的“括号”是怎么写的? val words = List("foo", "bar", "baz") val phrase = "These are upper case: " + words ma
我只是想知道,对于以下代码,编译器是否单独使用关联性/优先级或其他一些逻辑来评估。 int i = 0, k = 0; i = k++; 如果我们根据关联性和优先级进行评估,postfix ++具有比
我设置了一个 Azure FrontDoor 服务,以主/备份类型的方式将流量分配给两个 API 管理服务。就像我希望所有流量都流向我的主要 APIM 服务一样,如果我碰巧关闭该服务(假装中断),那么
这是一个简单的 CSS: /* Smartphones (portrait and landscape) ----------- */ @media only screen and (min-devi
我设置了一个 Azure FrontDoor 服务,以主/备份类型的方式将流量分配给两个 API 管理服务。就像我希望所有流量都流向我的主要 APIM 服务一样,如果我碰巧关闭该服务(假装中断),那么
来自 Programming Perl pg 90,他说: @ary = (1, 3, sort 4, 2); print @ary; 排序右侧的逗号在排序之前求值,而左侧的逗号在排序之
+----+------------+------+ | id | title | lang | +----+------------+------+ | 1 | title 1 EN |
如何使用 Java 获取 DiffServe 代码点 (DSCP) 整数的优先级部分?我预计它涉及位移位,但由于某种原因,我似乎无法获得我期望的值。 最佳答案 假设我理解正确,只需向右执行 3 位逻辑
我有下一个运行良好的 js 函数: $(function () { $(".country").click(function () { var countries = Arra
int a[3]={10,20,30}; int* p = a; cout << *p++ << endl; 根据 wikipedia ,后缀++的优先级高于解引用,*p++应该先运行p++再解引用结
我想在优先读取归档后解决这种类型的表达式 2+3/5*9+3-4 这是我尝试解决该任务的代码我该如何解决这个问题 while ( !inputFile.eof() ) { getline( inp
我正在玩 Rhino 并注意到这种奇怪的行为似乎是运算符优先级: js> {}+{} NaN js> ''+{}+{} [object Object][object Object] js> ''+({
我想遍历文件列表并检查它们是否存在,如果文件不存在则给出错误并退出。我写了下面的代码: FILES=( file1.txt file2.txt file3.txt ) for file in ${FI
我正在执行级联 SELECT: SELECT * FROM x WHERE a = 1 AND b = 2 AND c = 3 => If nothing found, try: SELECT * F
即将参加考试,我正在参加之前的考试。 问题: 当两个或多个样式表规则应用于同一元素时,以下哪种类型的规则将优先? 一个。任何来自浏览器的声明 b.有用户来源的正常声明 C。作者来源正常声明 d.文档级
我是一名优秀的程序员,十分优秀!