C/C++多参数函数参数的计算顺序与压栈顺序的示例代码

 更新时间:2020年06月01日 09:30:02   作者:_程序兔  
这篇文章主要介绍了C/C++多参数函数参数的计算顺序与压栈顺序,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

一、前言

今天在看Thinking in C++这本书时,书中的一个例子引起了我的注意,具体是使用了下面这句

单看这条语句的语义会发现仅仅是使用一个简单的string的substr函数将所得子串push_back到strings。但是在阅读时我却对于substr的参数传递产生了疑惑,到底是先执行了++current,还是先执行了last-current?

经过查阅资料,发现了两个相关知识点----参数的计算顺序与压栈顺序。

二、参数压栈顺序

C/C++中规定了函数参数的压栈顺序是从右至左,对于含有不定参数的printf函数,其原型是printf(const char* format,…);其中format确定了printf的参数(通过format的%个数判断)。假设是从左至右压栈,那么先入栈的是format(这里我们简化理解为参数个数),然后依次入栈未知参数,此时想要知道参数个数,就必须找到format,而要找到format,就必须知道参数个数,陷入一个逻辑矛盾。因此C/C++中规定参数压栈为从右至左,这样对于不定参数,最后入栈的是参数个数,只需要取栈顶就可以得到。可以通过下面的程序验证:

#include <stdio.h>
void foo(int x, int y, int z)
{
    printf("x = %d at [%X]\n", x, &x);
    printf("y = %d at [%X]\n", y, &y);
    printf("z = %d at [%X]\n", z, &z);
}
int main(int argc, char *argv[])
{
    foo(100, 200, 300);
    return 0;
}

 

通过输出结果可以看到x,y,z的栈内地址依次是 x < y < z;而栈的生长方向是从高到低,也就是先入栈的占高地址,因此z先入栈,其次是y,最后是x,即压栈顺序从右至左。

三、参数计算顺序

知道参数压栈顺序从右至左,是不是可以得出结论strings.push_back( s.substr(++current, last-current)); 先执行last-current,再执行++current呢?其实不然,先执行哪个参数和参数的计算顺序有关,而 C/C++中没有规定函数参数的计算顺序,即计算顺序依照编译器 ,编译器规定从右至左计算就先执行last-current,规定从左至右就先执行++current,笔者试过codeblocks与vscode的计算顺序都是从右至左。

也正因为函数参数的计算顺序依照编译器的实现,因此,C/C++的代码编写中并不支持编写诸如 func(++x, x+y)这种的程序,在不同编译器下可能产生不同的结果,所以上述代码应该分开写为:

 int len = last - current;
    ++current;
    strings.push_back(
      s.substr(current, len));

总结

到此这篇关于C/C++多参数函数参数的计算顺序与压栈顺序的示例代码的文章就介绍到这了,更多相关c/c++函数参数压栈顺序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++ 将文件数据一次性加载进内存实例代码

    C++ 将文件数据一次性加载进内存实例代码

    这篇文章主要介绍了C++ 将文件数据一次性加载进内存实例代码的相关资料,需要的朋友可以参考下
    2017-05-05
  • C++报错:Id returned 1exit status的解决办法

    C++报错:Id returned 1exit status的解决办法

    最近刚学c语言,不止一次遇到了同一种报错,经过总结分享给大家,下面这篇文章主要给大家介绍了关于C++报错:Id returned 1exit status的解决办法,需要的朋友可以参考下
    2023-04-04
  • C++实现万年历源代码

    C++实现万年历源代码

    这篇文章主要介绍了C++实现万年历源代码,可以直接在VC6.0编译运行,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • C语言实现Fibonacci数列递归

    C语言实现Fibonacci数列递归

    这篇文章主要介绍了C语言实现Fibonacci数列递归,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-02-02
  • 详解C++中String类模拟实现以及深拷贝浅拷贝

    详解C++中String类模拟实现以及深拷贝浅拷贝

    这篇文章主要介绍了详解C++中String类模拟实现以及深拷贝浅拷贝的相关资料,希望通过本文能帮助到大家,让大家实现这样的方法,需要的朋友可以参考下
    2017-10-10
  • C语言中强制类型转换的常见方法

    C语言中强制类型转换的常见方法

    强制类型转换是一种将一个数据类型转换为另一个数据类型的方法,这篇文章主要为大家整理了C语言中强制类型转换的方法,需要的可以参考一下
    2023-05-05
  • C++ Boost Fusion创建异构容器详解

    C++ Boost Fusion创建异构容器详解

    Boost.Fusion 使创建异构容器成为可能。例如,您可以创建一个向量,其第一个元素是 int,第二个元素是字符串。此外,Boost.Fusion 提供了处理异构容器的算法。您可以将 Boost.Fusion 视为异构容器的标准库
    2022-11-11
  • 快速学习六大排序算法

    快速学习六大排序算法

    这篇文章主要介绍了六大排序算法-插入排序、希尔排序、选择排序、冒泡排序、堆排序、快速排序,需要学习的小伙伴可以参考这篇文章
    2021-08-08
  • STL容器之list源码详细解读

    STL容器之list源码详细解读

    这篇文章主要介绍了STL容器之list源码详细解读,相对于vector的连续线性空间,list就显得更加复杂,它每插入或者删除一个元素,就配置或释放一个元素空间,需要的朋友可以参考下
    2024-01-01
  • 详解C语言的void*空指针

    详解C语言的void*空指针

    这篇文章主要为大家详细介绍了C语言的void*空指针,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03

最新评论