C语言的函数返回值和指针

简介: C|函数返回值(区分各类值)和指针(区分各类存储空间)的细节

我们知道,C的函数调用通常是通过栈机制来实现的。函数调用时,会发生地址的跳转,在跳转之前,通过存储需要返回的地址,后存储的地址先返回,便可以实现逐个函数调用,逐地址返回。每个函数使用一个栈帧空间(用寄存器ebp和esp来标识这个空间的起始位置和结束位置)来存储寄存器状态、返回地址、参数、和函数的局部变量。当返回时,esp降低到ebp位置,以便这一段空间可以被其它调用的函数重复使用。

所以,我们通常会说,返回局部变量的地址是错误的操作。

但值可以正常返回,原因是值可以通过寄存器和浮点寄存器返回。当返回的数据量比较大时,如超过两个字长的字节,两个寄存器都放不下,通常的做法是在被调用函数的栈帧段规划出一段内存,将返回值复制到这段空间。

include

int test1()
{//代码效果参考:http://www.zidongmutanji.com/bxxx/521640.html

int a = 8;
return a; // 返回值暂存在寄存器

}
double test2()
{
double a = 8.8;
return a; // 返回值暂存在寄存器
}
struct St{
int arr[111];
};
struct St test3()
{
struct St st = {0};
return st;
};
main()
{
int a = test1(); // mov eax,dword ptr [ebp-4]
double d = test2(); // fld qword ptr [ebp-8]
struct St st = test3();
getchar();
}
看下面的汇编代码:

22: int a = test1();
0040113E call @ILT+0(test1) (00401005)
00401143 mov dword ptr [ebp-4],eax // 值返回
23: double d = test2();
00401146 call @ILT+5(test2) (0040100a)
0040114B fstp qword ptr [ebp-0Ch] // 值返回
24: struct St st = test3();
0040114E lea eax,[ebp-540h]
00401154 push eax // 用于存放返回值的地址(被调函数的栈帧上)压栈
00401155 call @ILT+10(test3) (0040100f)
0040115A add esp,4
0040115D mov esi,eax
0040115F mov ecx,6Fh
00401164 lea edi,[ebp-384h]
0040116A rep movs dword ptr [edi],dword ptr [esi]
0040116C mov ecx,6Fh
00401171 lea esi,[ebp-384h]
00401177 lea edi,[ebp-1C8h]
0040117D rep movs dword ptr [edi],dword ptr [esi]
25: getchar();
————————————————
18: return st;
004010E7 mov ecx,6Fh
004010EC lea esi,[ebp-1BCh]
004010F2 mov edi,dword ptr [ebp+8]
004010F5 rep movs dword ptr [edi],dword ptr [esi]
004010F7 mov eax,dword ptr [ebp+8]
返回被调函数(callee function)的局部空间的内存空间的地址当然是不对的,但主调函数(caller function)的栈空间的地址还是可以返回的。

先看返回被调函数局部变量的地址的情形:

include

int demo()
{
int a = 5;
return &a;
}
main()
{
int
p = demo();
printf("%d\n",*p);// debug模式下输出5
getchar();
}
在Debug编译模式下,发现也可以正常输出变量的值?why?原因是这段可以覆盖的空间暂时没有被覆盖,那段空间,那个值还在,称为历史值(系统不会去操心这些历史值,如清零或置其它值,因为这些操作都会产生overhead)。Debug编译模式相对于Release,前者为了便于调试,会保留大量的调试信息,且不做优化。在Release模式下,上面的代码会生成一个随机值。

以下代码调用一个函数去覆盖前一次函数调用使用的栈帧空间:
//代码效果参考:http://www.zidongmutanji.com/zsjx/391099.html

include

int demo()
{
int a = 5;
return &a;
}
void foo()
{
int arr[1000]={0};
}
main()
{
int
p = demo();
foo(); // foo会覆盖掉demo使用的空间
//printf("abc"); // 调用库内函数是同样的效果
printf("%d\n",*p); // debug模式下输出0
getchar();
}
返回被调函数的栈空间地址是可以的:

include

int demo(int p)
{
*p = 88;
return p;
}

main()
{
int a;
int p = demo(&a);
printf("%d\n",
p); // 88
getchar();
}
以下三类空间的地址是可以返回的:

include

include

int g_var = 66;
int test1()
{
g_var++;
return &g_var; // 返回全局变量的地址,通常是没必要这样操作的
}
int
test2()
{
static s_var = 55;
return &s_var; // 返回局部静态变量的地址
}

int test3()
{
int
p = (int)malloc(sizeof(int));
return p; // 返回堆空间地址
}
main()
{
int
p = test1();
p = test2();
p = test3();
getchar();
}

相关文章
|
16天前
|
存储 C语言
【C语言基础】一篇文章搞懂指针的基本使用
本文介绍了指针的概念及其在编程中的应用。指针本质上是内存地址,通过指针变量存储并间接访问内存中的值。定义指针变量的基本格式为 `基类型 *指针变量名`。取地址操作符`&`用于获取变量地址,取值操作符`*`用于获取地址对应的数据。指针的应用场景包括传递变量地址以实现在函数间修改值,以及通过对指针进行偏移来访问数组元素等。此外,还介绍了如何使用`malloc`动态申请堆内存,并需手动释放。
|
16天前
|
存储 Serverless C语言
【C语言基础考研向】11 gets函数与puts函数及str系列字符串操作函数
本文介绍了C语言中的`gets`和`puts`函数,`gets`用于从标准输入读取字符串直至换行符,并自动添加字符串结束标志`\0`。`puts`则用于向标准输出打印字符串并自动换行。此外,文章还详细讲解了`str`系列字符串操作函数,包括统计字符串长度的`strlen`、复制字符串的`strcpy`、比较字符串的`strcmp`以及拼接字符串的`strcat`。通过示例代码展示了这些函数的具体应用及注意事项。
|
19天前
|
存储 C语言
C语言程序设计核心详解 第十章:位运算和c语言文件操作详解_文件操作函数
本文详细介绍了C语言中的位运算和文件操作。位运算包括按位与、或、异或、取反、左移和右移等六种运算符及其复合赋值运算符,每种运算符的功能和应用场景都有具体说明。文件操作部分则涵盖了文件的概念、分类、文件类型指针、文件的打开与关闭、读写操作及当前读写位置的调整等内容,提供了丰富的示例帮助理解。通过对本文的学习,读者可以全面掌握C语言中的位运算和文件处理技术。
|
19天前
|
存储 C语言
C语言程序设计核心详解 第七章 函数和预编译命令
本章介绍C语言中的函数定义与使用,以及预编译命令。主要内容包括函数的定义格式、调用方式和示例分析。C程序结构分为`main()`单框架或多子函数框架。函数不能嵌套定义但可互相调用。变量具有类型、作用范围和存储类别三种属性,其中作用范围分为局部和全局。预编译命令包括文件包含和宏定义,宏定义分为无参和带参两种形式。此外,还介绍了变量的存储类别及其特点。通过实例详细解析了函数调用过程及宏定义的应用。
|
19天前
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
|
24天前
|
Linux C语言
C语言 多进程编程(三)信号处理方式和自定义处理函数
本文详细介绍了Linux系统中进程间通信的关键机制——信号。首先解释了信号作为一种异步通知机制的特点及其主要来源,接着列举了常见的信号类型及其定义。文章进一步探讨了信号的处理流程和Linux中处理信号的方式,包括忽略信号、捕捉信号以及执行默认操作。此外,通过具体示例演示了如何创建子进程并通过信号进行控制。最后,讲解了如何通过`signal`函数自定义信号处理函数,并提供了完整的示例代码,展示了父子进程之间通过信号进行通信的过程。
|
24天前
|
C语言
C语言 字符串操作函数
本文档详细介绍了多个常用的字符串操作函数,包括 `strlen`、`strcpy`、`strncpy`、`strcat`、`strncat`、`strcmp`、`strncpy`、`sprintf`、`itoa`、`strchr`、`strspn`、`strcspn`、`strstr` 和 `strtok`。每个函数均提供了语法说明、参数解释、返回值描述及示例代码。此外,还给出了部分函数的自实现版本,帮助读者深入理解其工作原理。通过这些函数,可以轻松地进行字符串长度计算、复制、连接、比较等操作。
|
24天前
|
存储 安全 C语言
C语言 二级指针应用场景
本文介绍了二级指针在 C 语言中的应用,
|
25天前
|
SQL 关系型数据库 C语言
PostgreSQL SQL扩展 ---- C语言函数(三)
可以用C(或者与C兼容,比如C++)语言编写用户自定义函数(User-defined functions)。这些函数被编译到动态可加载目标文件(也称为共享库)中并被守护进程加载到服务中。“C语言函数”与“内部函数”的区别就在于动态加载这个特性,二者的实际编码约定本质上是相同的(因此,标准的内部函数库为用户自定义C语言函数提供了丰富的示例代码)
|
27天前
|
Linux
在Linux内核中根据函数指针输出函数名称
在Linux内核中根据函数指针输出函数名称