- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在为我的一个项目编写一个简单的行为模型引擎。我现在一直在尝试制作一个特定的函数 constexpr。
问题在底部,其余是背景信息
行为节点可以返回三种状态之一:
enum class State { Fail, Success, Running };
默认叶节点总是返回这三种状态之一:
constexpr inline auto success() { return []{ return State::Success; }; }
constexpr inline auto fail() { return []{ return State::Fail; }; }
constexpr inline auto running() { return []{ return State::Running; }; }
可以执行不同的操作,例如取反:
constexpr inline auto operator~(auto rule) {
return [rule]{
const auto result = rule();
return result == State::Fail ? State::Success :
result == State::Success ? State::Fail :
State::Running;
};
}
...和其他逻辑操作:
constexpr inline auto operator&&(auto left, auto right) {
return [left, right]{
const auto lresult = left();
const auto rresult = right();
return (lresult == State::Fail || rresult == State::Fail) ? State::Fail :
(lresult == State::Success && rresult == State::Success) ? State::Success :
State::Running;
};
}
最后是评价函数,很简单:
inline State execute(auto rule) {
return rule();
}
CHECK( State::Success == execute(success() && success()) );
CHECK( State::Fail == execute(success() && fail()) );
编写此示例时我无法访问 C++17 编译器,因此无法实际编译和测试它。
在这里,我们混合使用可能在编译时评估的规则,以及一些在运行时评估的规则。一个家伙走到一扇锁着的门前并试图打开它,如果他有 key 是在编译时决定的。
auto walking = false;
auto door_open = false;
auto is_at_door = false;
constexpr auto dude_has_key = false;
constexpr auto unlock_door = []{return []{
return dude_has_key ? State::Success : State::Fail;
};};
const auto walk_to_door = [&]{return []{
if(walking) { // Stop at the door
walking = false;
is_at_door = true;
return State::Success;
}
if(!is_at_door) {
walking = true;
return State::Running;
}
return State::Fail; // Cannot walk through the door
};};
const auto open_door = [&]{return []{
// We want to compile-time check if the door is locked
// hence we never check for that condition here.
if(!door_open) {
door_open = true;
}
return State::Success
};};
const auto rules = sequence({walk_to_door, unlock_door && open_door});
CHECK( State::Running == execute(rules) ); // Walking to door
CHECK( State::Fail == execute(rules) ); // Cannot unlock door
编写此示例时我无法访问 C++17 编译器,因此无法实际编译和测试它。
struct MyThing {
static constexpr bool PropertyA = true;
static constexpr int PowerLevel = 42;
};
struct YourThing {
static constexpr bool PropertyA = false;
static constexpr int PowerLevel = 43;
};
constexpr auto has_property_a(auto thing) {
return []{ return decltype(thing)::PropertyA ? State::Success :: State::Fail; };
}
constexpr auto has_minimum_power_level(auto thing, auto min_pl) {
return []{ return decltype(thing)::PowerLevel > min_pl ? Satet::Success ::State::Fail; };
}
template<typename ThingA, typename ThingB>
constexpr auto rule = sequence({
has_property_a(ThingA{}),
has_minimum_power_level(ThingB{}, 9000)
});
execute(sequence({rule<MyThing,YourThing>, __some_runtime_rule});
...是应该通过以下测试的排序函数
CHECK( State::Success == execute(sequence({success(), success(), success()})) );
CHECK( State::Fail == execute(sequence({success(), success(), fail()})) );
CHECK( State::Running == execute(sequence({success(), success(), running()})) );
// last (fail) never evaluated due to 2nd being running
CHECK( State::Running == execute(sequence({success(), running(), fail()})) );
换句话说,当序列中的所有节点都返回 State::Success
时,它返回 State::Success
。如果遇到 State::Running
,它会在那里“等待”,直到 State::Success
或 State::Fail
。 State::Fail
使整个序列失败。
这个函数的当前 make-the-tests-pass-implementation 看起来像
using Rule = std::function<State()>;
inline auto sequence(std::initializer_list<Rule> rules) {
return [rules]{
auto result = State::Success;
for(auto next : rules) {
const auto next_result = next();
result = (result == State::Success && next_result == State::Success) ? State::Success :
(result == State::Success && next_result == State::Running) ? State::Running :
State::Fail;
if(result == State::Fail || result == State::Running) {
return result;
}
}
return State::Success;
};
}
现在我该如何制作这个 constexpr?
最佳答案
如果你的序列只是为了重复应用&&,那么你可以用折叠来指定它
允许空的sequence()
(返回等同于success
)
template <typename ... Rules>
constexpr auto sequence(Rules ... rules)
{
return [rules...](){ return State::Success && ... && rules(); };
}
或者要求至少一条规则
template <typename First, typename ... Rules>
constexpr auto sequence(First first, Rules ... rules)
{
return [first, rules...](){ return first() && ... && rules(); };
}
关于c++ - 构造行为排序函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43674431/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!