在JavaScript或c#中,是否有一种方法可以确定逻辑表达式的结果所有变量的值已检索?
或者换句话说;是否可以对表达式进行评估,使其返回"true","false"或"maybe"?其中"maybe"表示需要更多的数据。
解释:我有一个过程,其中需要一些时间从数据库后端检索数据,我想看看如果没有必要,我们是否可以跳过检索某些数据。但逻辑表达式是预先确定的,不能更改或拆开。
例如,考虑以下表达式:
((a = 1) and (b = 2)) or (c = 3)
有几种可能性:如果a
和b
已被检索,但c
尚未被检索:
- 如果
a=1
和b=2
,那么表达式将始终返回true,我可以跳过检索c
的值 - 如果
a=0
和b=2
,那么第一部分将为假,我需要检索c
的值,以便能够确定结果
如果c
已经被检索到,而a
和b
还没有被检索到。
- 如果
c=3
那么表达式将始终返回true,我可以跳过检索a
和b
的值。 - 如果
c=2
,那么第一部分将为假,我需要检索a
和b
的值,以便能够确定结果
在这些情况下,只要知道结果已经确定,或者需要更多的数据,就可以大大加快进程。
有谁有主意吗?一个过程,函数,算法?
因此,为了覆盖您的特定代码,您可以简单地使用以下代码:
if(CHasValue())
return (c == 3) or ((a == 1) and (b == 2))
else
return ((a == 1) and (b == 2)) or (c == 3)
短路操作人员将负责其余部分。
对于更复杂的表达式,这不是很好地扩展。为了真正覆盖任意的布尔表达式,你需要创建自己的新类型和相应的布尔运算符。
我们将从定义一个布尔值的接口开始,该接口可能还没有计算出它的值:
public interface IComputableBoolean
{
public bool Value { get; }
public bool ValueComputed { get; }
}
第一个实现是简单的;它是一个可计算的布尔值,表示一个我们已知的值:
public class ComputedBoolean : IComputableBoolean
{
public ComputedBoolean(bool value)
{
Value = value;
}
public bool Value { get; private set; }
public bool ValueComputed { get { return true; } }
}
然后是稍微复杂一点的情况,基于函数生成的布尔值(可能是可能长时间运行的东西,或者有副作用)。它将接受一个计算表达式的委托,在第一次请求值时对其求值,并从那时起返回一个缓存的值(并指示它已经计算了它的值)。
public class DeferredBoolean : IComputableBoolean
{
private Func<bool> generator;
private bool? value = null;
public DeferredBoolean(Func<bool> generator)
{
this.generator = generator;
}
public bool Value
{
get
{
if (value != null)
return value.Value;
else
{
value = generator();
return value.Value;
}
}
}
public bool ValueComputed { get { return value != null; } }
}
现在我们只需要创建应用于该接口的And
、Or
和Not
方法。它们应该首先检查是否计算了足够的值以允许它短路,如果没有,它们应该创建一个延迟的布尔值来表示值的计算。这种延迟值的传播很重要,因为它允许组合复杂的布尔表达式,并且仍然可以用最少的计算量适当地短路。
public static IComputableBoolean And(
this IComputableBoolean first,
IComputableBoolean second)
{
if (first.ValueComputed && !first.Value ||
second.ValueComputed && !second.Value)
return new ComputedBoolean(false);
else
return new DeferredBoolean(() => first.Value && second.Value);
}
public static IComputableBoolean Or(
this IComputableBoolean first,
IComputableBoolean second)
{
if (first.ValueComputed && first.Value ||
second.ValueComputed && second.Value)
return new ComputedBoolean(true);
else
return new DeferredBoolean(() => first.Value && second.Value);
}
Not
操作有点不同,因为它根本不能短路,但它仍然很重要,因为它继续延迟给定布尔表达式的求值,因为它可能最终不需要。
public static IComputableBoolean Not(
this IComputableBoolean boolean)
{
if (boolean.ValueComputed)
return new ComputedBoolean(boolean.Value);
else
return new DeferredBoolean(() => boolean.Value);
}
我们现在可以像这样表示表达式(根据需要使用实际的长时间运行的操作来计算a
, b
和/或c
):
var a = new DeferredBoolean(() => false);
var b = new DeferredBoolean(() => true);
var c = new DeferredBoolean(() => false);
var expression = a.And(b).Or(c);
bool result = expression.Value;
假设a、b、c在加载后为真,而在加载前为假(或者a、b、c包含一个可以检查的"loaded"属性):
var isValid = false;
if (a && b) {
if (a == 1 && b == 2) {
isValid = true;
}
}
if (!isValid && c) {
if (c == 3) {
isValid = true;
}
}