0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

使用 MISRA C++:2023® 避免基于范围的 for 循环中的错误

北汇信息POLELINK 2024-03-28 13:53 次阅读

MISRA C++:2023®,MISRAC++标准的下一个版本,就在这里!为了帮助您了解MISRA C++:2023与上一版本之间的变化,我们继续Perforce首席技术支持工程师Frank van den Beuken博士撰写的第三期博客系列。

在前两篇博客中,我们向您介绍了新的 MISRA C++ 标准和C++ 的历史。在这篇博客中,我们将仔细研究以 C++ 中for循环为中心的特定规则。

什么是 MISRA C++:2023Rule9.5.2,为什么它很重要?

MISRA C++:2023引入了规则9.5.2,“ for范围初始值设定项 最多应包含一个函数调用”,以避免在基于范围的for语句的for范围初始值设定项创建临时对象时可能发生的未定义行为。

为了理解为什么会发生这种情况,让我们仔细看看基于 C++ 范围的for循环。

什么是 C++ 中基于范围的for循环?

编程中,循环用于重复代码块。当您知道要在代码块中循环多少次时,请使用for循环。

C++ 基于范围的for循环是在C++11中引入的,作为容器迭代的简洁表示法。

传统循环源自 C 语言,具有可选的循环初始化,然后是循环条件,最后是循环增量表达式。

传统for循环可用于迭代容器,如下所示:

std::vector v = { "Example", "vector", "of", "strings" };
for ( auto &&i = v.begin(); i != v.end(); ++i ) {
std::cout << *i << “ “; 
}
std::cout << std::endl; 

使用基于范围的for时,迭代器的使用是隐式的:

for ( auto &&s: v ) {
std::cout << s << “ “; 
}

对于同一循环,这是一个更简单的表示法。C++ 语言标准指出它是以下方面的缩写:

{
auto && __range = v;
auto __begin = __range;
auto __end = v.end();

for (; __begin != __end; ++__begin) {
auto &&s = *__begin;
std::cout << s << “ “; 
}
}

但是,这种表示法存在一定的局限性。在上面的示例中, __range 是用v初始化的,这是一个更简单的变量,但也可以使用一个复杂的表达式,为其创建多个临时对象。

让我们考虑使用一个函数,该函数返回字符串的向量,并具有:

  • 一个输出用空格分隔的字符串的循环,如上所述
  • 第二个循环,打印第一个字符串的字母,用空格分隔:
std::vector createStrings() {
return { "Example”, "vector", "of", "strings" };
}
int main() {
for ( auto w: createStrings() ) { std::cout << w << " "; } 
std::cout << std::endl; 
for ( auto c: createStrings()[0] ) { std::cout << c << " "; } 
std::cout << std::endl; 
}

如果我们执行此操作,第一个循环将按预期运行,但第二个循环将调用未定义的行为。 问题是 createStrings()[0]有两个函数调用。最里面的调用是createStrings的调用 ,最外面的调用是对索引运算符[]的调用。

未定义行为的原因是 “ createStrings ”返回的临时对象 用作“ operator[ ]”调用的参数,因此,根据C++的规则,临时对象的生存期不会延长。

返回页首

MISRA C++:2023Rule9.5.2 如何防范未定义的行为

MISRA C++:2023Rule9.5.2 旨在防止这种情况。MISRA C++:2023引入了规则9.5.2,该规则 要求for范围初始值设定项最多应包含一个函数调用。

它还建议通过在循环范围之前的单独声明中执行内部函数调用来解决此问题。例如:

auto strings = createStrings();
for ( auto c: strings[0] ) { std::cout << c << " "; } 

现在,初始值设定项中只有一个函数调用,因此生存期扩展具有所需的效果,并且行为已完全定义。

请注意,此问题已在 C++23 中得到解决,其中初始值设定项的所有临时项的生存期已扩展到整个for语句。

使用 Helix QAC 执行MISRA C++:2023规则

Perforce 的Helix QAC是一种静态分析工具,在提供 MISRA C 和MISRA C++合规性检查以及许多其他有价值的分析功能方面处于领先地位。

Helix QAC 通过其标准合规性模块为MISRA C++:2023规则提供100%的强制执行覆盖率,现已推出。静态分析工具查找并报告C和C++中违反MISRA规则和指令的情况。

欢迎联系北汇信息,申请Helix QAC试用。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • C语言
    +关注

    关注

    180

    文章

    7530

    浏览量

    128701
  • 代码
    +关注

    关注

    30

    文章

    4555

    浏览量

    66771
  • MISRA
    +关注

    关注

    0

    文章

    18

    浏览量

    6887
收藏 人收藏

    评论

    相关推荐

    MISRA C 2012 学习资料

    `MISRA C 学习资料 `
    发表于 09-17 14:55

    如何停止while循环中的for循环,停止计时?

    各位大师,小弟请教如何在运行计时后,停止计时;也就是说,如何停止while循环中的for循环,谢谢大师们!
    发表于 10-23 16:29

    如何在while循环中持续TCP监听?

    刚开始接触Labview,要做一个小项目, 功能是从TCP协议获取某个IP的数据,然后本地保存,然后写入本机某个端口等待其他上位机读写。问题是 我在While循环中放入TCP监听,当我用小工具模拟
    发表于 02-04 23:46

    避免在同一个循环中放置两个事件结构

    建议在同一个循环中,只放置一个事件结构。此时,当一个事件发生时,事件结构将对事件进行处理,然后继续循环,事件结构再等待下一个事件发生。如在同一个循环中放置两个事件结构,只有在两个事件结构都处理
    发表于 03-17 11:19

    同一事件在不同的循环中发生,为什么只要进入别的循环就会直接动作?怎样避免

    本帖最后由 我拿吉他弹棉花 于 2017-5-29 15:10 编辑 比如按钮就是,一个按钮在五个不同的循环中发生同样的事件。只要进入另外一个循环就直接发生了!怎么避免。点一次就在本循环
    发表于 05-02 15:21

    无限循环中的程序错误

    你好,我的节目刚开始就开始了。我开始初始化,右跳到无限循环中。氰去甲肾上腺素CysISR(InDebug
    发表于 08-16 06:57

    C++在嵌入式应用中的机遇与挑战是什么?

    C++在嵌入式应用中的机遇与挑战是什么?什么是MISRA C++
    发表于 04-28 06:25

    条件断点循环中的技巧

    一、条件断点循环中经常用到这个技巧,比如:遍历1个大List的过程中,想让断点停在某个特定值。参考上图,在断点的位置,右击断点旁边的小红点,会出来一个界面,在Condition这里填入断...
    发表于 01-27 06:59

    无法从ifinite循环中恢复ESP是怎么回事?

    我检查了我能找到的所有信息,但我无法从 ifinite 循环中恢复我的 ESP。我应该在什么地方闪烁什么来打破这个循环?将来我将只在启动过程中使用 init.lua 以避免像现在这样的情况,但现在我需要恢复我的设备。它在 wif
    发表于 04-26 06:58

    Arduino可将对话混合在无限循环中

    电子发烧友网站提供《Arduino可将对话混合在无限循环中.zip》资料免费下载
    发表于 11-30 11:29 0次下载
    Arduino可将对话混合在无限<b class='flag-5'>循环中</b>

    西门子博途SCL: FOR:在计数循环中执行

    使用“在计数循环中执行”指令,重复执行程序循环,直至运行变量不在指定的取值范围内。
    的头像 发表于 07-18 10:07 5522次阅读

    静态代码分析器工具Helix QAC 2023.2: 提供 100% 的 MISRA C:2012 和 MISRA C:2023 规则覆盖率

    Helix QAC 2023.2 提供 100% 的 MISRA C:2012 和 MISRA C:2023 规则覆盖率,并更新相应的合规性模块以支持 MISRA C:
    的头像 发表于 07-31 22:53 772次阅读
    静态代码分析器工具Helix QAC 2023.2: 提供 100% 的 <b class='flag-5'>MISRA</b> C:2012 和 <b class='flag-5'>MISRA</b> C:<b class='flag-5'>2023</b> 规则覆盖率

    您需要了解的有关下一个MISRA®标准的信息:MISRA C++ 2023®简介

    MISRA C++2023®是广受期待的MISRA C++ ®标准的下一个版本,将于今年晚些时候发布。新版本将整合AUTOSAR C++1
    的头像 发表于 08-25 18:06 932次阅读
    您需要了解的有关下一个<b class='flag-5'>MISRA</b>®标准的信息:<b class='flag-5'>MISRA</b> <b class='flag-5'>C++</b> <b class='flag-5'>2023</b>®简介

    静态代码分析工具Helix QAC 2023.3:将100%覆盖MISRA C++:2023®规则

    Helix QAC 2023.3预计将于2023年第四季度发布的新MISRA C++®指南,将100%覆盖MISRA C++:
    的头像 发表于 11-08 18:37 226次阅读
    静态代码分析工具Helix QAC 2023.3:将100%覆盖<b class='flag-5'>MISRA</b> <b class='flag-5'>C++</b>:<b class='flag-5'>2023</b>®规则

    for循环中i++与++i的区别

    i++和++i都是增加变量i的值的运算符,但它们之间有着一些微小的区别。在这篇文章中,我将详尽、详实、细致地解释i++和++i之间的差异,并探讨它们在循环中的应用。 首先,让我们明确i++
    的头像 发表于 11-26 09:20 1297次阅读