Fixed bug surrounding implies with parentheses when unnecessary.

Fixed correct saving of operation in elimination of implication.

Must use macro on helpers
This commit is contained in:
Martin Berg Alstad 2024-06-25 19:57:23 +02:00
parent 23cbb83391
commit 21f66d9409
4 changed files with 35 additions and 17 deletions

View File

@ -93,7 +93,7 @@ impl Display for Expression {
format!("{}{}", fmt_helper(left, Some(expression)), fmt_helper(right, Some(expression)))
}
Expression::Binary { left, operator: BinaryOperator::Or, right } => {
if parent.is_none() || matches!(parent, Some(Expression::Binary { operator: BinaryOperator::Or, .. })) {
if parent.is_none() || matches!(parent, Some(Expression::Binary { operator: BinaryOperator::Or | BinaryOperator::Implication, .. })) {
format!("{} {}", fmt_helper(left, Some(expression)), fmt_helper(right, Some(expression)))
} else {
format!("({} {})", fmt_helper(left, Some(expression)), fmt_helper(right, Some(expression)))

View File

@ -3,6 +3,7 @@ use crate::expressions::expression::Expression;
use crate::expressions::operator::BinaryOperator;
#[inline]
#[must_use]
pub fn and<L, R>(left: L, right: R) -> Expression
where
L: Into<Rc<Expression>>,
@ -12,6 +13,7 @@ where
}
#[inline]
#[must_use]
pub fn or<L, R>(left: L, right: R) -> Expression
where
L: Into<Rc<Expression>>,
@ -21,6 +23,7 @@ where
}
#[inline]
#[must_use]
pub fn implies<L, R>(left: L, right: R) -> Expression
where
L: Into<Rc<Expression>>,
@ -30,6 +33,7 @@ where
}
#[inline]
#[must_use]
pub fn binary<L, R>(left: L, operator: BinaryOperator, right: R) -> Expression
where
L: Into<Rc<Expression>>,
@ -39,11 +43,13 @@ where
}
#[inline]
#[must_use]
pub fn not<T: Into<Rc<Expression>>>(value: T) -> Expression {
Expression::Not(value.into())
}
#[inline]
#[must_use]
pub fn atomic<T: Into<String>>(value: T) -> Expression {
Expression::Atomic(value.into())
}

View File

@ -10,6 +10,7 @@ use crate::routing::response::Operation;
#[derive(Debug, PartialEq, Serialize)]
#[serde(rename_all = "SCREAMING_SNAKE_CASE")]
#[allow(clippy::enum_variant_names)]
pub enum Law {
EliminationOfImplication,
DeMorgansLaws,
@ -86,24 +87,27 @@ impl Expression {
/// Eliminate the implication operator from the expression.
/// This is done by replacing `a ➔ b` with `¬a b`.
fn elimination_of_implication(&self, operations: &mut Vec<Operation>) -> Self {
let result = match self {
Expression::Not(expr) => not(expr.elimination_of_implication(operations)),
match self {
Expression::Not(expr) => {
not(expr.elimination_of_implication(operations))
}
Expression::Binary { left, operator, right } => {
let l_result = left.elimination_of_implication(operations);
let r_result = right.elimination_of_implication(operations);
let before = binary(l_result.clone(), *operator, r_result.clone());
if let BinaryOperator::Implication = *operator {
or(not(l_result), r_result)
let after = or(not(l_result.clone()), r_result.clone());
if let Some(operation) = Operation::new(&before, &after, Law::EliminationOfImplication) {
operations.push(operation);
}
after
} else {
binary(l_result, *operator, r_result)
before
}
}
atomic @ Expression::Atomic(_) => atomic.clone(),
};
if let Some(operation) = Operation::new(self, &result, Law::EliminationOfImplication) {
operations.push(operation);
}
result
}
/// Eliminate double negations from the expression.
@ -112,10 +116,16 @@ impl Expression {
fn double_negation_elimination(&self, operations: &mut Vec<Operation>) -> Self {
let result = match self {
Expression::Not(expr) => {
let before = not(expr.double_negation_elimination(operations));
if let Expression::Not(inner) = expr.deref() {
inner.double_negation_elimination(operations)
let after = inner.double_negation_elimination(operations);
dbg!(before.to_string(), after.to_string());
if let Some(operation) = Operation::new(&before, &after, Law::DoubleNegationElimination) {
operations.push(operation);
}
after
} else {
not(expr.double_negation_elimination(operations))
before
}
}
Expression::Binary { left, operator, right } => {
@ -125,9 +135,9 @@ impl Expression {
}
atomic @ Expression::Atomic(_) => atomic.clone(),
};
if let Some(operation) = Operation::new(self, &result, Law::DoubleNegationElimination) {
operations.push(operation);
}
// if let Some(operation) = Operation::new(self, &result, Law::DoubleNegationElimination) {
// operations.push(operation);
// }
result
}
@ -285,7 +295,7 @@ mod tests {
assert_eq!(operations[0].before, "b ➔ c");
assert_eq!(operations[0].after, "¬b c");
assert_eq!(operations[1].law, Law::EliminationOfImplication);
assert_eq!(operations[1].before, "a ➔ b ➔ c");
assert_eq!(operations[1].before, "a ➔ ¬b c");
assert_eq!(operations[1].after, "¬a ¬b c");
}
@ -294,6 +304,7 @@ mod tests {
let mut operations = vec![];
let expression = and(atomic("a"), atomic("b")).elimination_of_implication(&mut operations);
assert_eq!(expression, and(atomic("a"), atomic("b")));
assert_eq!(operations.len(), 0);
}
#[test]
@ -301,6 +312,7 @@ mod tests {
let mut operations = vec![];
let expression = or(atomic("a"), and(atomic("b"), atomic("c"))).elimination_of_implication(&mut operations);
assert_eq!(expression, or(atomic("a"), and(atomic("b"), atomic("c"))));
assert_eq!(operations.len(), 0);
}
#[test]
@ -336,7 +348,7 @@ mod tests {
assert_eq!(operations[0].after, "¬a");
assert_eq!(operations[1].law, Law::DoubleNegationElimination);
assert_eq!(operations[1].before, "¬¬¬¬¬a");
assert_eq!(operations[1].after, "¬a");
assert_eq!(operations[1].after, "¬¬¬a");
}
#[test]

View File

@ -501,7 +501,7 @@ mod tests {
fn test_complex_expression() {
let expression = implies(and(atomic("A"), atomic("B")), or(atomic("C"), atomic("D")));
let header = TruthTable::extract_header(&expression);
assert_eq!(header, vec!["A", "B", "A ⋀ B", "C", "D", "C D", "A ⋀ B ➔ (C D)"]);
assert_eq!(header, vec!["A", "B", "A ⋀ B", "C", "D", "C D", "A ⋀ B ➔ C D"]);
}
#[test]