前言 这篇文章介绍C语言的内联函数、递归函数、函数指针、指针函数、局部地址、const关键字、extern关键字等知识点;这些知识点在实际项目开发中非常常用,非常重要。...函数返回局部空间的地址问题 子函数: 在调用结束后空间会被释放—被系统回收。 总结:子函数不能返回局部变量的地址。...内联函数 内联函数: 在调用的时候不会进行压栈出栈(不会经历保存地址的过程和恢复地址的过程)。 内联函数相当于一个替换的过程。 内联函数设计要注意:内联函数里只能写简单的代码—不能写复杂代码。...递归函数 什么是递归函数? 子函数直接或者间接的方式调用自己的过程叫做递归。 函数自己调用自己的过程—递归。 递归函数注意事项:必须有终止条件。...return 0; } int func(int a) { a++; if(a==10) { return a; } else { func(a); } } 示例2:计算字符串的长度—使用递归
直接调用 // 直接调用 add 函数 , 运行该函数 // 函数名 add 就是函数地址 add(1, 2); 2、定义函数指针变量 如果不使用 add 函数名 调用函数 , 使用 函数类型的指针...函数类型 int (int, int) , 定义为 func_add 名称 , 使用时需要使用该类型的指针调用函数 , 也就是 func_add* 类型 ; // int (int, int) 函数类型重命名为...func_add // 使用时需要使用该类型的指针调用函数 typedef int fun_add(int, int); 使用定义的 函数类型 调用函数 : 定义函数类型 的 指针类型 func_add...; 定义函数类型示例 : 将指向 int add(int x, int y) 函数的 函数指针类型 int (*)(int, int) , 定义为 pFun_add名称 , 使用时需要使用该类型变量调用函数...pFun_add)(int, int); 使用定义的 函数指针类型 调用函数 : 定义函数指针类型 pFun_add 的 变量 , 然后 将 add 函数地址 赋值给 该变量 , 然后 通过 函数指针变量
一、函数指针做函数参数 1、使用函数指针间接调用函数 在上一篇博客 【C++】函数指针 ③ ( 函数指针语法 | 函数名直接调用函数 | 定义函数指针变量 | 使用 typedef 定义函数类型 | 使用...typedef 定义函数指针类型 ) 中 , 最后一个示例 , 使用 typedef 定义函数指针类型 如下 : // int (int, int) 函数指针类型重命名为 pFun_add // 该类型变量可以直接接收..., 使用传入的函数进行计算 int caculate(pFun_add fun, int x, int y) { printf("使用 fun 函数计算 x 和 y = %d\n", fun(x, y...其它函数中执行 ; 4、函数指针做参数意义 函数指针做参数意义 : 提高程序灵活性 : 通过使用函数指针作函数参数 , 这样可以 在 程序 运行时 动态地 设置 要调用的函数 , 提高了程序的灵活性...定义为 结构体 的一部分 , 并使用该 结构体 来传递具有特定行为的对象的地址 ; 该操作有助于更好地组织代码 , 使代码更易于理解和维护 ; 错误处理 : 使用函数指针 , 将错误处理函数作为参数传递给其他函数
/**输入2个整数,然后让用户选择1或2,选1时调用max函数,输出2者中的大数, 选2时调用min函数,输出2者中的小数**/ #include #include <stdlib.h
注: 使用库函数,必须包含 #include 对应的头文件。 如何学会使用库函数?...,通过函数改变外部变量时就使用传址调用。...函数的嵌套调用和链式访问 函数和函数之间可以根据实际的需求进行组合的,也就是互相调用的。 6.1 嵌套调用 函数可以嵌套调用,但是不能嵌套定义。...那如何解决上述的问题: 将递归改写成非递归。 使用static对象替代 nonstatic 局部对象。...在递归函数设计中,可以使用 static 对象替代 nonstatic 局部对象(即栈对象),这不仅可以减少每次递归调用和返回时产生和释放 nonstatic 对象的开销,而且 static 对象还可以保存递归调用的中间状态
函数指针 1.1 定义 函数指针是指向函数的指针。它存储了函数的地址,允许在运行时动态地选择要调用的函数。函数指针允许在代码中间接地调用不同的函数。它是一个指针,可以指向函数,并用于调用该函数。...int (*add)(int, int); // 定义一个指向参数为两个整数、返回类型为整数的函数的指针 void (*func)(void); // 定义一个指向无参数、无返回值的函数的指针...而不是add=addNumbers(int a, int b) ; int result = add(3, 4);//等价于int result = (* add)(3, 4); // 使用函数指针调用函数...指针函数 2.1 定义 指针函数指的是一个函数,其返回类型是一个指针。这种函数返回一个指针值,而不是基本类型的值。指针函数本身是函数,其返回值是一个指针。...两者区别 函数指针是指向函数的指针,允许在运行时选择调用的函数。 指针函数是一个函数,其返回类型是一个指针,它返回指针值而不是基本类型的值。
递归 递归就是一个函数在它的函数体内调用它自身。执行递归函数将反复调用其自身,每调用一次就进入新的一层。递归函数必须有结束条件。...注: 递归的时候,每次调用一个函数,计算机都会为这个函数分配新的空间,这就是说,当被调函数返回的时候,调用函数中的变量依然会保持原先的值,否则也不可能实现反向输出。...特点: 递归函数特点 每一级函数调用时都有自己的变量,但是函数代码并不会得到复制,如计算5的阶乘时每递推一次变量都不同; 每次调用都会有一次返回,如计算5的阶乘时每递推一次都返回进行下一次; 递归函数中...,位于递归调用前的语句和各级被调用函数具有相同的执行顺序; 递归函数中,位于递归调用后的语句的执行顺序和各个被调用函数的顺序相反; 递归函数中必须有终止语句。...2.用户栈 是用户进程空间中的一块区域,用于保存用户进程的子程序间相互调用的参数、返回值、返回点以及子程序(函数)的局部变量。 我们编写的递归程序属于用户程序,因此使用的是用户栈。
递归函数实在一个函数通过名字调用自身的情况下构成的。...num <= 1) { 14 return 1; 15 }else{ 16 return num * f(num - 1); 17 } 18 }); 这里使用的是命名函数表达式的方法实现递归...,将这个函数赋值给 factorial 。...这样即使在使用过程中对变量进行修改,也不会影响已赋值的递归函数进行调用,保证了代码的安全性。这种方式在严格模式和非严格模式下都适用。
如果一个函数在内部调用自身,这个函数就叫做递归函数 递归函数的简单定义如下: def recursion(): return recursion() 这只是一个简单的定义,什么也做不了。...,当然,我们需要能实际做事情的函数,有用的递归函数应该满足如下条件: (1)当函数直接返回值时有基本实例(最小可能性问题) (2)递归实例,包括一个或多个问题最小部分的递归调用 使用递归的关键在于将问题分解为小部分...,递归不能永远进行下去,因为它总是以最小可能性问题结束,而这些问题又存储在基本实例中。...理论上,所有递归函数都可以写成循环的方式,不过循环的逻辑不如递归清晰。 使用递归函数需要注意仿制栈溢出,在计算机中,函数调用通过栈(stack)这种数据结构实现的。...尾递归是指在函数返回时只能调用函数本身,return语句不能包含表达式,这样,编译器或解释器就可以对尾递归进行优化,使递归本身无论调用多少次都只占用一个栈帧,从而避免栈溢出的情况。
正文开始 字符指针变量 在指针的类型中我们知道有⼀种指针类型为字符指针 char* ; 一般使用: int main() { char ch = 'w'; char *pc = &ch; *pc...= 'w'; return 0; } 还有⼀种使用方式如下: int main() { const char* pstr = "hello world."...函数指针变量 什么是函数指针变量呢? 根据前⾯学习整型指针,数组指针的时候,我们的类⽐关系,我们不难得出结论: 函数指针变量应该是⽤来存放函数地址的,未来通过地址能够调⽤函数的。...int (*) (int x, int y) //pf3函数指针变量的类型 函数指针变量的使用 #include int Add(int x, int y) { return...len - 1; j++) //单次平移 { src[j] = src[j + 1]; } src[j] = tmp; } } 改进一: 这个思路当然可以,但是一次一次转毕竟太麻烦,就不能一次到位么
不过在使用指针函数时,需要注意一点,相信细心地读者已经发现了,对比func_sum和func_sum2函数,除了返回值不一样之外,还有一个不同的地方在于,在func_sum中,变量sum使用的是静态局部变量...因此,在使用指针函数的时候,一定要避免出现返回局部变量指针的情况。 那么为什么用了static就可以避免这个问题呢? 原因是一旦使用了static去修饰变量,那么该变量就变成了静态变量。...然后对函数指针进行初始化,前面已经提到过了,直接将函数名赋值给函数指针变量名即可。 上述代码运行的结果如下: 调用的时候,既可以直接使用函数指针调用,也可以通过函数指针所指向的值去调用。...(*p)所代表的就是函数指针所指向的值,也就是函数本身,这样调用自然不会有问题。有兴趣的同学可以去试一试。 为什么要使用函数指针? 那么,有不少人就觉得,本来很简单的函数调用,搞那么复杂干什么?...,就可以使用指针函数来实现,我们只需要修改函数指针初始化的地方,而不需要去修改每个调用的地方(特别是当调用特别频繁的时候)。
① 函数指针 ② 指针函数 ③ 数组指针 ④ 指针数组 还有个找抽的:⑤ 函数指针数组 经翻阅小学五年级语文课本得知,一个短语中的最后部分,是这个短语的中语,其余部分是定语(修饰语)。...也就是说,以上短语相当于: ① 指向函数的指针 ② 返回指针的函数 ③ 指向数组的指针 ④ 放了很多指针的数组 ⑤ 放了很多指向函数的指针的数组 ? 语文课上完了,然鹅,没用!...你不能在源码里写这些中文,我们需要将它们表达为代码。...p = &func // 让 p 指向函数 func ② 指针函数 func char * func(int, float); // 一个返回指针的函数 ③ 数组指针 q int num[10]; /...char* 指针的数组 ⑤ 函数指针数组 fs void func(double, int); // 一个普通函数 void (*fs[3])(double, int); // 一个存放了 3 个指向以上函数的指针的数组
如果一个函数在内部调用自身本身,则该函数就是递归函数 递归优缺点 优点:使用递归函数的优点是逻辑简单清晰 理论上,所有的递归函数都可以写成循环的方式,但循环的逻辑不如递归清晰... 缺点:过深的调用会导致栈溢出 栈溢出 使用递归函数需要注意防止栈溢出 在计算机中,函数调用是通过栈(stack)这种数据结构实现的 每当进入一个函数调用,栈就会加一层栈帧...,每当函数返回,栈就会减一层栈帧 由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出 尾递归 解决递归调用栈溢出的方法是通过尾递归优化 事实上尾递归和循环的效果是一样的...,所以,把循环看成是一种特殊的尾递归函数也是可以的 尾递归是指,在函数返回的时候,调用自身本身,并且return语句不能包含表达式 例如,def fun(n) : retrun n*fun(n-...尾递归事实上和循环是等价的,没有循环语句的编程语言只能通过尾递归实现循环 Python标准的解释器没有针对尾递归做优化,任何递归函数都存在栈溢出的问题 使用示例: def fact(n): return
指针函数 指针函数就是返回指针值的函数,本质是一个函数。所以指针函数等价于“返回值为指针的函数”。...//使用函数指针名调用,func = max system("pause"); } 上例给出了函数指针的两种最普遍的调用形式,一个是直接用定义的指针变量(* func)调用,一个是用指针名调用,...后者看起来就像是将原函数使用了新名称去调用一样,可以看成函数的别名。...再次强调一下:指针变量名和指针符号 * 一定要用括号包裹起来。函数指针还有一种常用的使用方式,就是使用typedef定义后再使用。...; //使用函数指针名调用,func = max system("pause"); } 3.
,在嵌入式行业使用较多,但是无论是PC软件还是嵌入式软件,理解函数指针的定义和使用,对于理解程序设计都是很有好处的。...函数指针的赋值 函数指针和其他指针一样定义之后使用之前也是需要初始化。...函数指针的用法 我们使用指针的时候,需要通过钥匙 * 来取其指向的内存里面的值,函数指针使用也如此。通过用(*pf)取出存在这个地址上的函数,然后调用它。...int *p=max;指针定义错误,改为:int (*p)(int ,int)=max; 解析:函数的指针是不能直接赋值给int型指针. scanf("%d %d %d",a,b,c);库函数使用错误,...d=p(p(a,b),c);函数指针调用函数错误,改为d=(*p)((*p)(a,b),c);` 解析:用函数指针调用函数的格式如下:(【*】【函数指针名称】)(【参数列表】);不能直接用函数指针加上参数就直接调用
,在嵌入式行业使用较多,但是无论是PC软件还是嵌入式软件,理解函数指针的定义和使用,对于理解程序设计都是很有好处的。...函数指针的赋值 函数指针和其他指针一样定义之后使用之前也是需要初始化。...函数指针的用法 我们使用指针的时候,需要通过钥匙 * 来取其指向的内存里面的值,函数指针使用也如此。通过用(*pf)取出存在这个地址上的函数,然后调用它。...int *p=max;指针定义错误,改为: int (*p)(int ,int)=max; 解析:函数的指针是不能直接赋值给int型指针. scanf("%d %d %d",a,b,c);库函数使用错误...d=p(p(a,b),c);函数指针调用函数错误,改为 d=(*p)((*p)(a,b),c);` 解析:用函数指针调用函数的格式如下:(【*】【函数指针名称】)(【参数列表】);不能直接用函数指针加上参数就直接调用
其中0x4013B0就是上图的func()函数的入口地址,从上图可以看到,func和&func的地址都一样,所以&对于函数而言,可以不需要 2.接下来便使用函数指针来指向上面func()函数 实例1如下...: #include "stdio.h" int func(int x) { return x; } int main() { int (*FP)(int); //定义FP函数指针...004013B0,004013B0 2)当使用typedef时,函数指针使用如下: #include "stdio.h" int func(int x) { return x;...} typedef int (*FP)(int); //使用声明函数指针类型FP int main() { FP a; //通过FP,定义函数指针a a=func;...再来定义函数指针: #include "stdio.h" int func(int x) { return x; } typedef int (FP)(int); //使用声明函数类型
javascript递归函数如何使用 说明 1、所有递归函数都有一个通用模式 。总是由一个调用自身的递归部分和一个不调用自身的基本情形组成。...2、当一个函数调用自己的时候,它就会将一个新的执行上下文推送到执行堆栈里。 这种情况会一直持续进行下去,直到遇到基本情形 ,然后堆栈逐个弹出展开成 各个上下文。...注意 依赖递归会导致运行时堆栈溢出错误。...for (let i = 2; i <= number; i++) { product *= i; } return product; }; 以上就是javascript递归函数的使用
printf("%d+%d", a, b); } void test() { //函数指针定义的三种方式 //1.先定义出函数类型,再通过函数类型定义指针变量 typedef void(FUNC_TYPE...//2.先定义出函数的指针类型,再通过类型定义函数的指针变量 typedef void(*FUNC_TYPE)(int, int); FUNC_TYPE func = fun1; func...//3.直接定义函数指针变量 void(*func)(int, int) = fun1; func(30,40); } int main() { test(); return 0; } ?...以上三种方式:第三种的使用率最高 函数指针和指针函数的区别 函数指针:指向函数的指针—void(*p)(int,int)=fun1; 指针函数:函数返回值是指针—int * fun1(int *a){return...a;} 函数指针数组-----》指针数组,数组里面每个元素类型都是函数指针 #define _CRT_SECURE_NO_WARNINGS #include void fun1()
add的地址输出 26 27 //定义函数指针 28 int(*pf)(int x, int y); 29 //函数指针名*pf两侧的括号不能省略,否则就成了一个返回“int...x = pf(3, 4);//通过函数指针pf调用函数add (使用函数指针不必像使用一般指针那样解引用) 35 int x = (*pf)(3, 4);//函数指针解引用 这样做的好处可以明确指明...同普通指针一样,如果 44 //没有明确的初始化,则函数指针的值将是一个随机数,使用这样的指针非常危险。...因此在使用函数指针之前对其进行初始化或着赋一个初值,即将一个函数名赋给 45 //该函数指针变量 46 cout << fun_ptr(7, 8)<<endl; 47 48 49...system("pause"); 50 return 0; 51 } 1 /* 2 3 指针函数的使用——回调函数 4 5 6 */ 7 #include<cstdlib
领取专属 10元无门槛券
手把手带您无忧上云