gpt4 book ai didi

java - 对于这样的事情,ANTLR 是正确的方法吗?

转载 作者:行者123 更新时间:2023-11-30 11:37:00 26 4
gpt4 key购买 nike

所以,我真的卡住了,开始认为这不是正确的方法。考虑这个字符串:“[APPLE|ORANGE] PEAR”这里的逻辑应该是“如果 APPLE 在字符串中,获取它的颜色 ELSE 如果 ORANGE 在字符串中,获取它的颜色 ELSE 返回空字符串。现在总是返回 PEAR 颜色”。

我能够得到 Hello World工作正常。1. 我正在努力让 ANTLR 理解我的字符串中的条件逻辑。2. 我正在努力让 ANTLR 为每个标识符调用 getColor()。任何帮助或指导将不胜感激。我现在气急败坏。

grammar Test;

@header {
package org.mytest.Test;
}

@members {
private String answer = "";

private void getColor(String fruit)
{
//Use java reflection to get Fruit class and invoke method getColor()
answer = fruit.color;
}

}

@lexer::header {
package org.mytest.Test;
}


/*------------------------------------------------------------------
* PARSER RULES
*------------------------------------------------------------------*/

row returns [List<String> list]
@init {list = new ArrayList<String>();}
: a=value {list.add($a.val);} (WS b=value {list.add($b.val);})* (EOF)
;

conditionalString returns [String color]:
(
a=IDENTIFIER (WS IDENTIFIER)* {getColor($a.text); }
)
{$color=answer;};

//Get the text string for the matched identifier?
value returns [String val] : IDENTIFIER {val = $IDENTIFIER.text;} ;


/*------------------------------------------------------------------
* LEXER RULES
*------------------------------------------------------------------*/

IDENTIFIER : ('A'..'Z')+;


WS : ( '\t' | ' ')+ { $channel = HIDDEN; } ;

理想的用法如下。 “上下文”参数将是一些资源句柄。我会手动编辑构造函数以传递上下文,以便 ANTL 知道如何处理每个标记。

TestLexer lex = new TestLexer(new ANTLRStringStream("[APPLE|ORANGE] PEAR"));
CommonTokenStream tok = new CommonTokenStream(lex);
TestParser par = new TestParser(context,tok);
System.out.println(par.conditionalString());

最佳答案

这是我认为接近您要求的语法。它只完全按照您指定的方式处理情况:[id|id|id] 是一个简单的条件条件,条件条件之外的任何 id 都按原样计算。

水果.g

grammar Fruit;

@parser::members {

private StringBuilder output = new StringBuilder();
private java.util.HashMap<String, String> colors = new java.util.HashMap<String, String>();

public void addColor(String fruit, String color){
colors.put(fruit, color);
}

private void printColor(String fruit){
if (colors.containsKey(fruit)){
output.append(colors.get(fruit));
output.append(" ");
} else {
output.append("(no color for ").append(fruit).append(")");
}
}

private void printColor(Token id){
printColor(id.getText());
}

private void evaluateCondition(java.util.List<Token> tokens){
for (Token token : tokens){
String fruit = token.getText();
if (colors.containsKey(fruit)){
printColor(fruit);
break;
}
}
}
}

conditionalString returns [String result]
@after { result = output.toString();}
: expr
;

expr : cond_expr+
;

cond_expr : ID
{printColor($ID);}
| LSQB values+=ID (OR values+=ID)* RSQB
{evaluateCondition($values);}
;

OR : '|';
LSQB : '[';
RSQB : ']';
ID : ('A'..'Z')+;
WS : ('\t'|' ')+ {skip();};

这是测试类。解析器上的方法 addColor 的存在是为了简化测试。例如,如果水果“ORANGE”的颜色为“orange”并且输入字符串为“ORANGE ORANGE”,则输出应为“orange orange”。

水果测试.java

public class FruitTest {

public static void main(String[] args) throws Exception {
CharStream input = new ANTLRStringStream("[APPLE|ORANGE] PEAR");
FruitLexer lexer = new FruitLexer(input);
CommonTokenStream tokens = new CommonTokenStream(lexer);

FruitParser parser = new FruitParser(tokens);

parser.addColor("APPLE", "red");
parser.addColor("ORANGE", "orange");
parser.addColor("PEAR", "yellow");

String result = parser.conditionalString();

if (lexer.getNumberOfSyntaxErrors() > 0 || parser.getNumberOfSyntaxErrors() > 0){
throw new Exception("Syntax errors encountered!");
}

System.out.println(result);
}
}

测试用例 1:很多水果

  • 苹果=红色
  • ORANGE = 橙色
  • 梨 = 黄色

输入:[APPLE|ORANGE] PEAR
输出:红黄

测试用例 2:大量水果

  • ORANGE = 橙色
  • 梨 = 黄色

输入:[APPLE|ORANGE] PEAR
输出:橙黄色

测试用例 3:水果含量低

  • 梨 = 黄色

输入:[APPLE|ORANGE] PEAR
输出:黄色

测试用例4:绝食(未定义水果颜色)

输入:[APPLE|ORANGE] PEAR
输出:(PEAR 没有颜色)

因为 PEAR 不是条件的一部分,所以它应该被定义。

关于java - 对于这样的事情,ANTLR 是正确的方法吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14270560/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com