- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有这段代码,用于在 Controller 的 LCD 显示器上构建图形界面;该代码使用 AVR 和 PIC32 针对 2 种不同的架构编译:
FishinoTftGuiLabel *l1;
FishinoTftGui
.Page(F("Page1"))
.Label(50, 140, 0, 24, LabelAlign::Left, F("Slider value:"))
.getElement(l1)
--
.Label(l1->x() + l1->w() + 10, 140, 0, 24, LabelAlign::Left, F("pippo"))
;
每个成员函数返回相同的对象(或相关的对象);例如,Label() 函数返回 FishinoTftGuiLabel 引用,可用于链接其他调用。getElement(T *&) 只是一种获取指向当前对象的指针的方法,它可以在后续调用中使用,而不会破坏链条并且不必为每个对象使用中间变量; -- 运算符返回到包含的 Page 对象。我的问题是“l1”指针应该在第一次创建标签时由 getElement 设置,它是在整个东西终止之后设置的,但只是在 AVR 平台上;在 PIC32 上评估正常。
因此,在 PIC32 上,顺序如下:
1) 评估第一个标签语句并创建标签
2) getElement(l1) 被执行,存储第一个标签的引用
3) 第二个Label语句被求值; l1->x() 正确使用了 Label1 引用
在AVR平台上,会出现这样的情况:
1) 所有 Label() 调用的所有参数首先被评估,因此 l1->x() 崩溃,因为调用了一个未初始化对象的成员
2) 接下来评估 Label() 函数
我的问题:这是一个编译器错误,还是在这种情况下不能保证链式调用之间的评估顺序?有没有一种方法可以强制执行正确的评估顺序,而不必在多个语句中破坏整个内容?
最佳答案
如评论中所述,在 C++17 之前,未指定函数参数中未排序子表达式的求值顺序,因此这不是编译器错误:允许两种顺序(如果这些表达式导致多次读/写同一个标量变量,例如 f(i++,i++)
).
自 C++17 起,后缀表达式(如函数调用)从左到右求值;函数参数的评估顺序仍未指定,但不能交错。因此,自 C++17 起,您的代码将始终提供所需的结果。
作为变通方法,您可以让 Label 和 friend 也接受 lambda 作为参数,以支持惰性(因此是有序的)评估,例如:
template<typename T>
auto constref_or_evaluate(T const& t) -> T const&
{ return t; }
template<typename T>
auto constref_or_evaluate(T&& t) -> decltype(std::forward<T>(t)())
{ return std::forward<T>(t)(); }
// the type of FishinoTftGui
struct FishinoTftGuiType
{
// chainable members ...
template<typename... T>
auto Label(T&&... t) -> FishinoTftGuiType&
{
LabelImpl( constref_or_evaluate(std::forward<T>(t))... );
return *this;
}
private:
// the original chainable member implementations ...
void LabelImpl(int,int); //whatever
};
// to be used as
FishinoTftGui
.Label(1,2)
.Label([]{return 3;},4);
此处,第二个 Label() 中的 lambda 将始终在第一个 Label() 完全计算后调用。
这还有一个好处是可以更好地控制惰性表达式的计算时间(例如,只要调整 View 大小时,标签就可以更新惰性参数,等等)。因此,在 >=C++17 代码中也值得考虑。
据我所知,这只是 C++11;无论如何,如果你还想传递 l/rvalue 引用参数,你需要编写一个 forward_or_evaluate()
函数;这在 C++11 中完全可行,但实现起来有点困难。
关于c++ - 链式函数中奇怪的评估顺序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48476469/
一晃五年没写博客了,依旧再C#上耕耘,依旧没有啥建树,现在也不知道.net上还有多少人再使用,在这里分享一些自己觉得写的还算优雅的代码。 对于自己写着完的代码,我特别喜欢链式(来源于jQuer
我正在构建一个吉他和弦查找应用程序。我使用多维数组来表示指板。数组中的每个元素都由具有字符串属性“Note”的 FretSpace 结构表示。为了初始化指板上的音符属性,我传递了要处理的吉他弦的详细信
我在演示代码中使用 setTimeout 函数模拟了 3 个 ajax 调用。我将从一段运行良好的代码开始:所有调用都是并行进行的,我希望所有调用都能成功,否则会出现错误。 var p1 = func
谁能解释一下? a = [2,3,4] b = [5,6,8,9] print(len(a) > 0) print(len(b) > 0) print((len(a) > 0) & len(b) >
我正在处理具有多个子 JSONObject 的 JSONObject。这是我填写内容的方式: myJson.getJSONObject(CAT_NAME).put(VAR_NAME, var)
想象一下这种情况,我有一个需要检查属性的对象。但是,该对象当前可以具有空值。 如何在一个“if”条件下检查这两个条件? 目前,我必须做这样的事情: if (myObject != null) {
我有一个对象集合,称它们为obj。他们有一个 act() 方法。 act() 方法最终会导致 o 上的 event() observable 调用 onComplete。 链接这些的好方法是什么? 即
假设我有一个列表变量 datalist 存储 10,000 个字符串实体。QTableView 只需要显示其中的一些实体。这就是为什么 QTableView 被指定为 QSortFilterProxy
我正在寻找支持链式 MSI 安装的工具(最好不是 InstallShield,而且最好是便宜/免费的)。我有几个小型安装需要能够单独部署,但也需要作为一个组部署,我不想维护多个安装程序。 看起来我需要
在这种情况下,我想迭代集合中除最后 2 个元素之外的所有元素。 假设我采用了一种奇怪的方式,例如 x.Reverse().Skip(2).Reverse()。 每个 LINQ 操作是否会有效地生成一个
对于javascript来说非常陌生,我有两个html数字选择,包括年份,我想将第二个选择与第一个选择链接起来,这样当我在第一个选择中选择年份时(而第二个选择没有选项)首先),第二个选择应包括从所选数
有人可以向我解释一下为什么以下两个链式函数: // returns zero if okay var resetCounter = function (model) { return new Prom
所以我有 2 个 promise 函数。当第一个函数出现错误时,我希望它显示错误消息。当完成或失败时,我希望他们执行一个finally catch all 函数,但由于某种原因它不起作用。我的代码如下
我有一个函数 const func = () => server.insertPatientSurveyQuestionToDataBase(Store.getPatientID(), SurveyN
(async function() { var a,b; function flush(){ return new Promise(res => {
这个问题已经有答案了: Promise chaining: Use result from previous promise in next then callback [duplicate] (1
这可能不是专业正则表达式理解的问题。唯一重要的是因为我正在运行多个链式替换命令,这些命令会影响文本文件中的某些相同文本。我还想象在替换之前,根据分隔符词(需要多次替换)的使用方式对 txt 文件进行分
我正在尝试构建一组类来定义 OSI 堆栈中协议(protocol)的分层属性...从抽象意义上讲,我只需要从父 python 类继承属性,但我需要能够调用整个类链一次...所以,我正在寻找这样的东西.
我正在努力兑现 promise ,到目前为止我偶然发现了这一点: new Promise((resolve, reject) => { setTimeout(() => { r
我试图理解 promise ,我需要链接它们并装饰来自不同端点的对象宽度数据。 例如: 我的 Node-express 应用程序中有这个 //controller.js export const ge
我是一名优秀的程序员,十分优秀!