- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
= r; -6ren">
我有像这样的中缀表达式:
((source = id)AND (target= id) AND ( NOT( color != blue) OR ( age<= 23 )))
如何转换为如下 TreeMap 所示的二叉树。我刚开始用 javascript 编写代码,但不知道如何编写插入操作,尤其是像 NOT 这样的一元运算符。如何解决这个问题。
var Node = function(data,left,right){
this.data = data;
this.left = left;
this.right = right;
};
var BinaryTree = function(){
this.head = new Node(null,null,null);
};
BinaryTree.prototype.insert = function(data){
if(this.head.data == null){
this.head = new Node(data,null,null);
} else{
var current = this.head;
insert(current,data);
}
};
var insert = function(current,data){
// How to insert ?
};
最佳答案
正如昨晚所说,这里是我必须要说的重点:
图节点:
//abstract base-class
class GraphNode {
constructor(){
Object.defineProperty(this, "parent", {
writable: true,
//enumerable: false, //so it doesn't show up in JSON
value: null
})
}
compute(ctx){ throw new Error("not implemented") }
toString(){ throw new Error("not implemented") }
}
//leaf-nodes
class ValueNode extends GraphNode{
constructor(value){
super();
this.value = value;
}
compute(){ return this.value; }
toString(){ return JSON.stringify(this.value); }
}
class PropertyNode extends GraphNode{
constructor(property){
super();
this.property = property;
}
compute(ctx){ return ctx[this.property]; }
toString(){ return String(this.property); }
}
//tree-nodes
class UnaryNode extends GraphNode{
constructor(op, node){
if(!(node instanceof GraphNode)){
throw new Error("invalid node passed")
}
super();
this.op = op;
this.node = node;
node.parent = this;
}
compute(ctx){
var v = this.node.compute(ctx);
switch(this.op){
case "NOT": return !v;
}
throw new Error("operator not implemented '"+this.op+"'");
}
toString(){
return "( " + this.op + " " + this.node.toString() + " )";
}
}
UnaryNode.operators = ["NOT"];
class BinaryNode extends GraphNode{
constructor(op, l, r){
if(!(l instanceof GraphNode && r instanceof GraphNode)){
throw new Error("invalid node passed")
}
super();
this.op = op;
this.left = l;
this.right = r;
l.parent = this;
r.parent = this;
}
compute(ctx){
var l = this.left.compute(ctx);
var r = this.right.compute(ctx);
switch(this.op){
//logic operators
case "AND": return l && r;
case "OR": return l || r;
//comparison-operators
case "=": return l === r;
case "<=": return l <= r;
case ">=": return l >= r;
case "!=": return l != r;
case ">": return l > r;
case "<": return l < r;
//computational operators
case "+": return l + r;
case "-": return l - r;
case "*": return l * r;
case "/": return l / r;
}
throw new Error("operator not implemented '"+this.op+"'");
}
toString(){
return "( " + this.left.toString() + " " + this.op + " " + this.right.toString() + " )";
}
}
//also defines precendence
BinaryNode.operators = [
"*","/","+","-",
">","<","<=",">=","!=","=",
"AND","OR",
]
//dot is kind of special:
class DotNode extends BinaryNode{
constructor(l, r){
/*
if(!(l instanceof PropertyNode || l instanceof DotNode)){
throw new Error("invalid left node")
}
*/
if(!(r instanceof PropertyNode)){
throw new Error("invalid right node")
}
super(".", l, r);
}
compute(ctx){
//especially because of this composition:
//fetch the right property in the context of the left result
return this.right.compute( this.left.compute(ctx) );
}
toString(){
return this.left.toString() + "." + this.right.toString();
}
}
解析器:
function escapeForRegex(str){
return String(str).replace(/[.*+?^=!:${}()|[\]\/\\]/g, '\\$&');
}
//dynamically build my parsing regex:
var tokenParser = new RegExp([
//numbers
/\d+(?:\.\d*)?|\.\d+/.source,
//string-literal
// /["](?:\\[\s\S]|[^"])+["]|['](?:\\[\s\S]|[^'])+[']/.source,
//booleans
//"true|false",
//operators
[".", "(", ")"].concat(UnaryNode.operators, BinaryNode.operators)
.sort((a,b) => b.length-a.length) //so that ">=" is added before "=" and ">", for example
.map(escapeForRegex)
.join("|"),
//properties
//has to be after the operators
/[a-zA-Z$_][a-zA-Z0-9$_]*/.source,
//remaining (non-whitespace-)chars, just in case
//has to be at the end
/\S/.source
].map(s => "("+ s +")").join("|"), "g");
function parse(str){
var tokens = [];
//abusing str.replace() as a RegExp.forEach
str.replace(tokenParser, function(token, number, op, property){
if(number){
token = new ValueNode(+number);
//}else if(string){
// token = new ValueNode(JSON.parse(string));
//}else if(bool){
// token = new ValueNode(bool === "true");
}else if(property){
token = new PropertyNode(property);
}else if(!op){
throw new Error("unexpected token '"+token+"'");
}
tokens.push(token);
});
for(var i; (i=tokens.indexOf(".")) > -1; ){
tokens.splice(i-1, 3, new DotNode(tokens[i-1], tokens[i+1]))
}
for(var i,j; (i=tokens.lastIndexOf("(")) > -1 && (j=tokens.indexOf(")", i)) > -1;){
tokens.splice(i, j+1-i, process(tokens.slice(i+1, j)));
}
if(~tokens.indexOf("(") || ~tokens.indexOf(")")){
throw new Error("mismatching brackets");
}
return process(tokens);
}
function process(tokens){
UnaryNode.operators.forEach(token => {
for(var i=-i; (i=tokens.indexOf(token, i+1)) > -1;){
tokens.splice(i, 2, new UnaryNode(token, tokens[i+1]));
}
})
BinaryNode.operators.forEach(token => {
for(var i=1; (i=tokens.indexOf(token, i-1)) > -1;){
tokens.splice(i-1, 3, new BinaryNode(token, tokens[i-1], tokens[i+1]));
}
});
if(tokens.length !== 1){
console.log("error: ", tokens.slice());
throw new Error("something went wrong");
}
return tokens[0];
}
及其用法:
var tree = parse("((a.source = id)AND (target= id) AND ( NOT( color != blue) OR ( age<= 23 )))")
//var tree = parse("1=1=age+10>30"); //to test operator precedence
var data = {
id: 12345,
a: { source: 12345 },
target: 12345,
color: "#FF0",
blue: "#00F",
age: 20
}
console.log(tree.compute(data));
console.log(tree.toString());
console.log(JSON.stringify(tree, null, 2));
关于javascript - 二叉表达式树的中缀,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37469768/
我正在用 yacc/bison 编写一个简单的计算器。 表达式的语法看起来有点像这样: expr : NUM | expr '+' expr { $$ = $1 + $3; } | expr '-'
我开始学习 lambda 表达式,并在以下情况下遇到了以下语句: interface MyNumber { double getValue(); } MyNumber number; nu
这两个 Linq 查询有什么区别: var result = ResultLists().Where( c=> c.code == "abc").FirstOrDefault(); // vs. va
如果我们查看 draft C++ standard 5.1.2 Lambda 表达式 段 2 说(强调我的 future ): The evaluation of a lambda-expressio
我使用的是 Mule 4.2.2 运行时、studio 7.5.1 和 Oracle JDK 1.8.0_251。 我在 java 代码中使用 Lambda 表达式,该表达式由 java Invoke
我是 XPath 的新手。我有网页的html源 http://london.craigslist.co.uk/com/1233708939.html 现在我想从上面的页面中提取以下数据 完整日期 电子
已关闭。这个问题是 off-topic 。目前不接受答案。 想要改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 已关闭10 年前。 Improve th
我将如何编写一个 Cron 表达式以在每天上午 8 点和下午 3:30 触发?我了解如何创建每天触发一次的表达式,而不是在多个设定时间触发。提前致谢 最佳答案 你应该只使用两行。 0 8 * * *
这个问题已经有答案了: What do 3 dots next to a parameter type mean in Java? (9 个回答) varargs and the '...' argu
我是 python 新手,在阅读 BeautifulSoup 教程时,我不明白这个表达式“[x for x in titles if x.findChildren()][:-1]”我不明白?你能解释一
(?:) 这是一个有效的 ruby 正则表达式,谁能告诉我它是什么意思? 谢谢 最佳答案 正如其他人所说,它被用作正则表达式的非捕获语法,但是,它也是正则表达式之外的有效 ruby 语法。 在
这个问题在这里已经有了答案: Why does ++[[]][+[]]+[+[]] return the string "10"? (10 个答案) 关闭 8 年前。 谁能帮我处理这个 JavaSc
这个问题在这里已经有了答案: What is the "-->" operator in C++? (29 个答案) Java: Prefix/postfix of increment/decrem
这个问题在这里已经有了答案: List comprehension vs. lambda + filter (16 个答案) 关闭 10 个月前。 我不确定我是否需要 lambda 或其他东西。但是,
C 中的 assert() 函数工作原理对我来说就像一片黑暗的森林。根据这里的答案https://stackoverflow.com/a/1571360 ,您可以使用以下构造将自定义消息输出到您的断言
在this页,John Barnes 写道: If the conditional expression is the argument of a type conversion then effec
我必须创建一个调度程序,它必须每周从第一天上午 9 点到第二天晚上 11 点 59 分运行 2 天(星期四和星期五)。为此,我需要提供一个 cron 表达式。 0-0 0-0 9-23 ? * THU
我正在尝试编写一个 Linq 表达式来检查派生类中的属性,但该列表由来自基类的成员组成。下面的示例代码。以“var list”开头的 Process 方法的第二行无法编译,但我不确定应该使用什么语法来
此 sed 表达式将输入字符串转换为两行输出字符串。两条输出行中的每一行都由输入的子串组成。第一行需要转换成大写: s:random_stuff\(choice1\|choice2\){\([^}]*
我正在使用 Quartz.Net 在我的应用程序中安排我的工作。我只是想知道是否可以为以下场景构建 CRON 表达式: Every second between 2:15AM and 5:20AM 最
我是一名优秀的程序员,十分优秀!