C语言-内联函数、递归函数、指针函数

简介: C语言-内联函数、递归函数、指针函数

1. 前言

这篇文章介绍C语言的内联函数、递归函数、函数指针、指针函数、局部地址、const关键字、extern关键字等知识点;这些知识点在实际项目开发中非常常用,非常重要。


下面就以小章节的流程介绍每个知识点。


2. 函数返回局部空间的地址问题

子函数: 在调用结束后空间会被释放—被系统回收。


总结:子函数不能返回局部变量的地址。


示例1:

#include <stdio.h>
char *func(void);
int main()
{
  printf("%s\n",func());  //打印不出来。
  return 0;
}
char *func(void)
{
  char buff[]="1234567890";
  return buff;
}

示例2:

#include <stdio.h>
char *func(char *p);
int main()
{
  char buff[]="1234567890";
  printf("%s\n",func(buff)); //可以打印
  return 0;
}
char *func(char *p)
{
  return p;
}

示例3:

#include <stdio.h>
char *func(void);
int main()
{
  printf("%s\n",func());  //可以打印
  return 0;
}
char *func(void)
{
  static char buff[]="1234567890";
  return buff;
}

3. const 只读关键字(常量)

(1) const关键字—修饰变量

#include <stdio.h>
int main()
{
  //const int a; //初始化不赋值,这行代码就没有意义
  const int a=100;
  a=200; //错误的代码--无法对常量赋值--只读变量赋值
  printf("a=%d\n",a);
  return 0;
}

(2) const关键字—修饰指针

#include <stdio.h>
//指针: 数据域、指针(地址)域
int main()
{
  int a=100;
  int b=200;
  //const常用4种修饰指针的方法
  const int *p1=&a; //指向空间值无法修改,指向的地址可以改变
  int const *p2=&a; //指向空间值无法修改,指向的地址可以改变
  int *const p3=&a; //指向空间值可以修改,指向的地址无法改变
  const int *const p4=&a;  //向空间值无法修改,指向的地址无法改变
  //int *p5 const; 语法是错误的
  //*p1=666; 错误的
  //p1=&b;   正确的
  //*p2=666;错误的
  //p2=&b;正确的
  //*p3=666;正确的
  //p3=&b;错误的
  //p4=&b;错误的
  //*p4=666;错误的
  return 0;
}

4. 内联函数

内联函数: 在调用的时候不会进行压栈出栈(不会经历保存地址的过程和恢复地址的过程)。

内联函数相当于一个替换的过程。

内联函数设计要注意:内联函数里只能写简单的代码—不能写复杂代码。

函数里的局部变量存放在栈空间里的。栈空间:是由系统管理的。

#include <stdio.h>
void func(void);
int main()
{
  int a;
  func();
  printf("12345\n");
  return 0;
}
//inline 声明-定义内联函数
inline void func(void)
{
  printf("hello\n");
}

5. extern 外部引用声明

extern 多用于多文件编程里变量、函数、其他数据类型的引用声明。

外部引用声明的时候,不能赋值。

#include <stdio.h>
//引用声明
extern int a;
extern char buff[];
extern void func();
int main()
{
  printf("%d\n",a);
  printf("%s\n",buff);
  func();
  return 0;
}
int a=100;
char buff[]="1234567890";
void func()
{
  printf("hello\n");
}

6. 字符串数组和字符串常量定义问题

#include <stdio.h>
int main()
{
  //p1指向的字符串是在程序编译的时候赋值
  char *p1="1234567890"; //指针指向字符串常量
  //p2数组是程序运行的时候赋值
  char p2[]="abcdefg";
  //p1[0]='A'; 错误的
  printf("%s\n",p1);
  printf("%s\n",p2);
  return 0;
}

示例2:

#include <stdio.h>
int main()
{
  //p1指向的字符串是在程序编译的时候赋值
  char *p1="1234567890"; //指针指向字符串常量
  //p2数组是程序运行的时候赋值
  char p2[]="abcdefg";
  int a;
  int b;
  int c;
  printf("a=%#x\n",&a);
  printf("b=%#x\n",&b);
  printf("c=%#x\n",&c);
  printf("p1=%#x\n",p1);
  printf("p2=%#x\n",p2);
  return 0;
}
/*
a=0xbf9f93e0
b=0xbf9f93dc
c=0xbf9f93d8
p1=0x8048524
p2=0xbf9f93e4
*/

7. 标准main函数形参语法

#include <stdio.h>
/*
int argc :传入的参数数量(包括可执行文件本身)
char **p :保存传入的数据地址
main传入的参数数据都是字符串类型。
传参数的方式:  ./a.out 123 456 789
*/
int main(int argc,char **p)
//int main(int argc,char *p[]) p[0] p[1] p[2]
{
  int i;
  for(i=0;i<argc;i++)
  {
    printf("p[%d]=%s\n",i,p[i]);
  }
  return 0;
}

8. 指针函数与函数指针

数组指针: 本身是指针,指向二维数组的指针(一维数组指针)。int (*p)[5];


指针数组: 本身是数组,数组里存放的是地址。int *p[5]; (相当于定义了5个指针)


数组的名称本身就是数组元素的首地址—数组名称就是地址。


**函数指针: **本身是指针,指向函数的指针。语法:int (*p)(int,int); 不支持++和—运算符。


指针函数: 本身是函数,表示函数的返回值是指针类型。语法: int *func(int a,int b){}


函数名称就是地址。


示例1:

#include <stdio.h>
int func(int a,int b);
int main(int argc,char **argv)
{
  int (*p)(int,int); //指向函数的指针
  p=func;
  printf("%d\n",func(10,20)); //通过函数名称调用函数
  printf("%d\n",p(10,20)); //通过指针调用函数--写法1
  printf("%d\n",(*p)(10,20)); //通过指针调用函数--写法2
  return 0;
}
int func(int a,int b)
{
  return a+b;
}

示例2: 函数指针当做函数形参

#include <stdio.h>
int func1(int a,int b);
int func2(int (*p)(int,int),int a,int b);
int main(int argc,char **argv)
{
  printf("%d\n",func2(func1,100,200));
  return 0;
}
int func1(int a,int b)
{
  return a+b;
}
int func2(int (*p)(int,int),int a,int b)
{
  return p(a,b);
}

9. 递归函数

什么是递归函数? 子函数直接或者间接的方式调用自己的过程叫做递归。

函数自己调用自己的过程—递归。

递归函数注意事项:必须有终止条件。

示例1:

#include <stdio.h>
int func(int a);
int main(int argc,char **argv)
{
  printf("%d\n",func(1)); //10
  return 0;
}
int func(int a)
{
  a++;
  if(a==10)
  {
    return a;
  }
  else
  {
    func(a);
  }
}

示例2:计算字符串的长度—使用递归

#include <stdio.h>
int func(char *p);
int main(int argc,char **argv)
{
  printf("%d\n",func("1234567890")); //10
  return 0;
}
//计算字符串长度
int func(char *p)
{
  if(*p=='\0')
  {
    return 0;
  }
  return 1+func(p+1);
}
/*
演示递归函数的返回过程:
a(); //3
int a()
{
  return 1+b();
}
int b()
{
  return 1+c();
}
int c()
{
  return 1;
}
*/
目录
相关文章
|
1月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
93 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
1月前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
70 10
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
53 9
|
1月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
59 9
|
1月前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
44 8
|
1月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
46 7
|
1月前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
53 6
|
1月前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
296 6
|
1月前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
67 6
|
1月前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
44 5