Diskuze: Parser matematických výrazů
V předchozím kvízu, Test znalostí C# .NET online, jsme si ověřili nabyté zkušenosti z kurzu.
Člen
Zobrazeno 12 zpráv z 12.
//= Settings::TRACKING_CODE_B ?> //= Settings::TRACKING_CODE ?>
V předchozím kvízu, Test znalostí C# .NET online, jsme si ověřili nabyté zkušenosti z kurzu.
Kód jsem proletěl velmi rychle a toto je první věc, které jsem si
všiml.
V metodách parseAddExp/parseMulExp nemáš lhs stejné jako
rhs.
Tedy v parseAddExpr máš lhs = parseMulExp(), ale rhs = parseAddExpr();. Nemělo by být rhs taky parseMulExp();?
Podobně pro parseMulExp...
Jinak podle mě lhs a rhs je trochu něco jiného a nazval bych to spíš jako levý (první) a pravý (druhý) operand.
U parseMulExp je ani stejné mít nemůžu, u parseAddExp jsem si toho teď všiml, ale nepamatuju si, proč jsem to tak napsal, takže to zkusím změnit a uvidím...
Aha tak jak jsem zjistil, když to změním, začne to padat s ParseError
Na kód jsem se nedíval, ale ta chyba by co popisuješ na začátku by se
dala vyřešit takto:
1 - 1 + 1 = 1 + (-1) + 1
Když pro levý i pravý operand použiješ stejnou funkci (parseAddExp/MultExp), a místo ifu použiješ while, tak by to mělo fungovat. Aspoň v mém kódu to tak funguje. Tady je ukázka z mého parseru (není to hotové, ale zatím to dělá co má) http://www.itnetwork.cz/dev-lighter/656
Je pravda, že ta gramatika nesedí úplně k tomu kódu, to jsem si teď všiml. Ale nedovedu ai to moc dobře představit jinak. Jak myslíš "stejnou funkci"? tomisoka no to je pravda, ale neřeší to můj problém
Zkus tento vstup 1 - 1 + 1 + 1. Vyjde ti z toho (1 - (1 + (1 + 1))). Správně by to mělo být naopak (((1 - 1) + 1) + 1).
parseAddExp:
IAstNode<int> lhs = parseMulExp();
while (stream.Match(TokenType.Add) || stream.Match(TokenType.Sub))
{
var tok = stream.Read();
var rhs = parseMulExp();
if (tok.Matches(TokenType.Add))
lhs = new AddNode(node, rhs);
else if (tok.Matches(TokenType.Sub))
lhs = new SubNode(node, rhs);
throw new ParseError("Strange behaviour, this exception shouldn't be thrown");
}
return lhs;
oprava: (nemuzu editovat)
IAstNode<int> lhs = parseMulExp();
while (stream.Match(TokenType.Add) || stream.Match(TokenType.Sub))
{
var tok = stream.Read();
var rhs = parseMulExp();
if (tok.Matches(TokenType.Add))
lhs = new AddNode(lhs, rhs);
else if (tok.Matches(TokenType.Sub))
lhs = new SubNode(lhs, rhs);
throw new ParseError("Strange behaviour, this exception shouldn't be thrown");
}
return lhs;
Nevidím teda rozdíl mezi těma dvěma verzema . No pravda je, že to hned vypadá líp s while a ne s rekurzí, jenže předtím mě to u volání mulExp v addExp rhs házelo chybu. Právěže vím kde mám chybu i jak by to mělo parsovat správně. Zkusím to hned jak budu na pc a uvidím... (u mulExp očekávam stejnou změnu)
Tak po změně mě to padá na "Strange behaviour"...
Nemůžu editovat, ale to bylo jen přehlédnutí, tahle vyjímka tam nemá co dělat už . Vypadá to, že to funguje
Zobrazeno 12 zpráv z 12.