namespace Canon.Core.GrammarParser; /// /// LR语法中的一个表达式,例如 'program_struct -> ~program_head ; program_body' /// 其中'~'标识当前移进到达的位置 /// public class Expression : IEquatable { /// /// 表达式的左部 /// public required NonTerminator Left { get; init; } /// /// 表达式的向前看字符串 /// public required Terminator LookAhead { get; init; } /// /// 表达式的右部 /// public required List Right { get; init; } /// /// 当前移进的位置 /// public int Pos { get; set; } public bool Equals(Expression? other) { if (other is null) { return false; } if (Right.Count != other.Right.Count) { return false; } for (int i = 0; i < Right.Count; i++) { if (Right[i].IsTerminated != other.Right[i].IsTerminated) { return false; } if (!Right[i].Equals(other.Right[i])) { return false; } } return Left == other.Left && LookAhead == other.LookAhead; } public override bool Equals(object? obj) { if (obj is not Expression other) { return false; } return Equals(other); } public override int GetHashCode() { int hash = Left.GetHashCode(); hash ^= LookAhead.GetHashCode(); foreach (TerminatorBase terminator in Right) { hash ^= terminator.GetHashCode(); } return hash; } public override string ToString() { string result = $"{Left} -> "; for (int i = 0; i < Right.Count; i++) { if (i == Pos) { result += '~'; } result += ' '; result += Right[i].ToString(); } result += $", {LookAhead}"; return result; } public static bool operator ==(Expression a, Expression b) { return a.Equals(b); } public static bool operator !=(Expression a, Expression b) { return !a.Equals(b); } }