ARTEMIS-3962 porting changes from AMQ-8613
Improve performance of selectors with a big sequence of OR and AND logical expressions.
This commit is contained in:
parent
f0ff385c7c
commit
f11b96e7ed
|
@ -16,78 +16,170 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.artemis.selector.filter;
|
package org.apache.activemq.artemis.selector.filter;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A filter performing a comparison of two objects
|
* A sequence of expressions, to be combined with OR or AND conjunctions.
|
||||||
*
|
*
|
||||||
* @version $Revision: 1.2 $
|
|
||||||
*/
|
*/
|
||||||
public abstract class LogicExpression extends BinaryExpression implements BooleanExpression {
|
public abstract class LogicExpression implements BooleanExpression {
|
||||||
|
|
||||||
|
protected final List<BooleanExpression> expressions = new ArrayList<>(2);
|
||||||
|
|
||||||
|
private LogicExpression(BooleanExpression lvalue, BooleanExpression rvalue) {
|
||||||
|
expressions.add(lvalue);
|
||||||
|
expressions.add(rvalue);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void addExpression(BooleanExpression expression) {
|
||||||
|
expressions.add(expression);
|
||||||
|
}
|
||||||
|
|
||||||
|
public BooleanExpression getLeft() {
|
||||||
|
if (expressions.size() == 2) {
|
||||||
|
return expressions.get(0);
|
||||||
|
}
|
||||||
|
throw new IllegalStateException("This expression is not binary: " + this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public BooleanExpression getRight() {
|
||||||
|
if (expressions.size() == 2) {
|
||||||
|
return expressions.get(1);
|
||||||
|
}
|
||||||
|
throw new IllegalStateException("This expression is not binary: " + this);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param left
|
* Returns the symbol that represents this binary expression. For example, addition is
|
||||||
* @param right
|
* represented by "+"
|
||||||
|
*
|
||||||
|
* @return
|
||||||
*/
|
*/
|
||||||
public LogicExpression(BooleanExpression left, BooleanExpression right) {
|
public abstract String getExpressionSymbol();
|
||||||
super(left, right);
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
if (expressions.size() == 2) {
|
||||||
|
return "( " + expressions.get(0) + " " + getExpressionSymbol() + " " + expressions.get(1) + " )";
|
||||||
|
}
|
||||||
|
StringBuilder result = new StringBuilder("(");
|
||||||
|
int count = 0;
|
||||||
|
for (BooleanExpression expression : expressions) {
|
||||||
|
if (count++ > 0) {
|
||||||
|
result.append(" " + getExpressionSymbol() + " ");
|
||||||
|
}
|
||||||
|
result.append(expression.toString());
|
||||||
|
}
|
||||||
|
result.append(")");
|
||||||
|
return result.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static BooleanExpression createOR(BooleanExpression lvalue, BooleanExpression rvalue) {
|
public static BooleanExpression createOR(BooleanExpression lvalue, BooleanExpression rvalue) {
|
||||||
return new LogicExpression(lvalue, rvalue) {
|
if (lvalue instanceof ORExpression) {
|
||||||
|
ORExpression orExpression = (ORExpression) lvalue;
|
||||||
@Override
|
orExpression.addExpression(rvalue);
|
||||||
public Object evaluate(Filterable message) throws FilterException {
|
return orExpression;
|
||||||
|
} else {
|
||||||
Boolean lv = (Boolean) left.evaluate(message);
|
return new ORExpression(lvalue, rvalue);
|
||||||
// Can we do an OR shortcut??
|
|
||||||
if (lv != null && lv.booleanValue()) {
|
|
||||||
return Boolean.TRUE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Boolean rv = (Boolean) right.evaluate(message);
|
|
||||||
return rv == null ? null : rv;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getExpressionSymbol() {
|
|
||||||
return "OR";
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static BooleanExpression createAND(BooleanExpression lvalue, BooleanExpression rvalue) {
|
public static BooleanExpression createAND(BooleanExpression lvalue, BooleanExpression rvalue) {
|
||||||
return new LogicExpression(lvalue, rvalue) {
|
if (lvalue instanceof ANDExpression) {
|
||||||
|
ANDExpression orExpression = (ANDExpression) lvalue;
|
||||||
@Override
|
orExpression.addExpression(rvalue);
|
||||||
public Object evaluate(Filterable message) throws FilterException {
|
return orExpression;
|
||||||
|
} else {
|
||||||
Boolean lv = (Boolean) left.evaluate(message);
|
return new ANDExpression(lvalue, rvalue);
|
||||||
|
|
||||||
// Can we do an AND shortcut??
|
|
||||||
if (lv == null) {
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
if (!lv.booleanValue()) {
|
|
||||||
return Boolean.FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
Boolean rv = (Boolean) right.evaluate(message);
|
|
||||||
return rv == null ? null : rv;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getExpressionSymbol() {
|
|
||||||
return "AND";
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public abstract Object evaluate(Filterable message) throws FilterException;
|
public abstract Object evaluate(Filterable message) throws FilterException;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean matches(Filterable message) throws FilterException {
|
public abstract boolean matches(Filterable message) throws FilterException;
|
||||||
Object object = evaluate(message);
|
|
||||||
return object == Boolean.TRUE;
|
private static class ORExpression extends LogicExpression {
|
||||||
|
|
||||||
|
ORExpression(BooleanExpression lvalue, BooleanExpression rvalue) {
|
||||||
|
super(lvalue, rvalue);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object evaluate(Filterable message) throws FilterException {
|
||||||
|
boolean someNulls = false;
|
||||||
|
for (BooleanExpression expression : expressions) {
|
||||||
|
Boolean lv = (Boolean)expression.evaluate(message);
|
||||||
|
if (lv != null && lv.booleanValue()) {
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
if (lv == null) {
|
||||||
|
someNulls = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (someNulls) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean matches(Filterable message) throws FilterException {
|
||||||
|
for (BooleanExpression expression : expressions) {
|
||||||
|
boolean lv = expression.matches(message);
|
||||||
|
if (lv) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getExpressionSymbol() {
|
||||||
|
return "OR";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class ANDExpression extends LogicExpression {
|
||||||
|
|
||||||
|
ANDExpression(BooleanExpression lvalue, BooleanExpression rvalue) {
|
||||||
|
super(lvalue, rvalue);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object evaluate(Filterable message) throws FilterException {
|
||||||
|
boolean someNulls = false;
|
||||||
|
for (BooleanExpression expression : expressions) {
|
||||||
|
Boolean lv = (Boolean)expression.evaluate(message);
|
||||||
|
if (lv != null && !lv.booleanValue()) {
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
if (lv == null) {
|
||||||
|
someNulls = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (someNulls) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean matches(Filterable message) throws FilterException {
|
||||||
|
for (BooleanExpression expression : expressions) {
|
||||||
|
boolean lv = expression.matches(message);
|
||||||
|
if (!lv) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getExpressionSymbol() {
|
||||||
|
return "AND";
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -126,7 +126,10 @@ public class SelectorTest {
|
||||||
|
|
||||||
assertSelector(message, "(trueProp OR falseProp) AND trueProp", true);
|
assertSelector(message, "(trueProp OR falseProp) AND trueProp", true);
|
||||||
assertSelector(message, "(trueProp OR falseProp) AND falseProp", false);
|
assertSelector(message, "(trueProp OR falseProp) AND falseProp", false);
|
||||||
|
assertSelector(message, "(falseProp OR falseProp OR falseProp OR falseProp OR falseProp OR falseProp OR trueProp)", true);
|
||||||
|
assertSelector(message, "(falseProp OR falseProp OR falseProp OR falseProp OR falseProp OR falseProp OR falseProp)", false);
|
||||||
|
assertSelector(message, "(trueProp AND trueProp AND trueProp AND trueProp AND trueProp AND trueProp AND falseProp)", false);
|
||||||
|
assertSelector(message, "(trueProp AND trueProp AND trueProp AND trueProp AND trueProp AND trueProp AND trueProp)", true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
Loading…
Reference in New Issue