11.C 语言学习:递归、宏定义、预处理、汉诺塔、Fibonacci 等

C 语言学习:递归、宏定义、预处理、汉诺塔、Fibonacci 等

1. 函数递归

什么是递归?

函数调用自己本身的行为称为递归。

递归本质上也是一种循环,但实现方式不同,特点是:

✔ 优点

  • 逻辑清晰、代码简短
  • 适合解决回溯、树形结构、数学归纳类问题(如汉诺塔、斐波那契)

❗ 必须满足两个条件

  1. 必须要有结束条件,否则会导致无限递归 → 栈溢出 Segmentation fault
  2. 避免递归层次过深

2. 字符串打印与 strlen()

通过递归前先学习字符串处理:

c 复制代码
void show_str(char a[])
{
    int len = strlen(a); // 字符串真实长度(不含'\0')
    for(int i=0; i<len; i++)
    {
        putchar(a[i]);
    }
    printf("\nstrlen %lu\n", strlen(a));
}

注意事项:

  • sizeof(str) 得到的是数组容量,例如 char str[100] → 100 字节
  • strlen(str) 得到的是字符串长度,例如 "hello" → 5

这是 C 语言中 字符串与字符数组的关键区别


3. 闰年判断函数

闰年规则:

  • 能被 4 整除但不能被 100 整除
  • 或能被 400 整除
c 复制代码
int isLeapYear(int year)
{
    return ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0);
}

用于遍历打印 1900 ~ 2024 之间的所有闰年:

c 复制代码
for(int i=1900; i<2025; i++)
{
    if(isLeapYear(i))
        printf("year is %d\n", i);
}

4. 递归的典型应用

4.1 阶乘计算

c 复制代码
int fun(int n)
{
    if(n == 1) return 1;
    return n * fun(n - 1);
}

递归非常适合数学归纳结构的函数表达式。


4.2 汉诺塔问题

问题描述

三根柱子 A、B、C,将 n 个盘从 A 移动到 C,每次只能移动一个盘,大盘不能压在小盘上。

c 复制代码
void haonuota(int n, char src, char tmp, char dst)
{
    if (n == 1)
    {
        printf("%c -> %c\n", src, dst);
    }
    else
    {
        haonuota(n - 1, src, dst, tmp);
        printf("%c -> %c\n", src, dst);
        haonuota(n - 1, tmp, src, dst);
    }
}

汉诺塔体现了递归的 "逆向思维":

  1. 先把 n-1 个盘移动到中间柱子
  2. 把最大盘移动到目标柱子
  3. 再把 n-1 个盘移到目标柱子

4.3 Fibonacci 数列

递归版:

c 复制代码
int fib(int n)
{
    if(n == 1 || n == 2)
        return 1;
    return fib(n-1) + fib(n-2);
}

但这种写法效率非常低(指数复杂度)。

推荐:循环版 Fibonacci(性能好)

c 复制代码
int fib_loop(int n)
{
    if (n == 1 || n == 2)
        return 1;

    int a = 1, b = 1, c = 0;

    for(int i = 3; i <= n; i++)
    {
        c = a + b;
        a = b;
        b = c;
    }
    return c;
}

5. 宏定义(

宏属于预处理阶段,特点是:

  • 代码替换,不会执行运算
  • 无类型检查
  • 效率高,没有函数调用开销

示例:

c 复制代码
#define PI 3.14
#define GETMAX(x,y) ((x)>(y)?(x):(y))
#define M 50
#define N (M+M)

带参宏建议:

所有参数加括号避免优先级错误,例如:

c 复制代码
#define GETMAX(x,y) ((x)>(y)?(x):(y))

5.1 多行宏

c 复制代码
#define PRINT(a,b,c) do{ \
    printf("%d\n", a);   \
    printf("%d\n", b);   \
    printf("%d\n", c);   \
}while(0)

6. C 语言预处理

编译步骤:

  1. 预处理 (处理 #include, #define,宏展开)
  2. 编译(变成汇编)
  3. 汇编(生成 .o 目标文件)
  4. 链接(生成最终 a.out 或可执行程序)

7. 头文件

#include <...>:从系统目录中查找,比如 /usr/include
#include "...":先从当前目录查找,再去系统目录查找

头文件一般包含:

  • 宏定义
  • typedef
  • 函数声明
  • extern 声明

防止头文件被重复包含

c 复制代码
#ifndef _FUNC_H_
#define _FUNC_H_

extern int add(int a, int b);

#endif

8. 条件编译

常见语法:

c 复制代码
#if
#elif
#else
#endif

判断宏是否存在:

c 复制代码
#ifdef DEBUG
#endif

#ifndef DEBUG
#endif

非常适合调试或不同平台切换。

相关推荐
少控科技16 小时前
QT新手日记024 - QT001程序代码
开发语言·qt
kaoa00020 小时前
Linux入门攻坚——62、memcached使用入门
linux·运维·memcached
莫非王土也非王臣20 小时前
深度学习之对比学习
人工智能·深度学习·学习
Wzx19801220 小时前
doker深学习
学习·docker
码农水水20 小时前
国家电网Java面试被问:TCP的BBR拥塞控制算法原理
java·开发语言·网络·分布式·面试·wpf
20130924162721 小时前
1968年 Hart, Nilsson, Raphael 《最小成本路径启发式确定的形式基础》A* 算法深度研究报告
人工智能·算法
如何原谅奋力过但无声21 小时前
【力扣-Python-滑动窗口经典题】567.字符串的排列 | 424.替换后的最长重复字符 | 76.最小覆盖子串
算法·leetcode
InterestOriented21 小时前
破解银发学习痛点 兴趣岛 “普惠 + 品质” 模式打造积极老龄化范本
大数据·人工智能·学习
model200521 小时前
alibaba linux3 系统盘清理
linux·运维·服务器
浮尘笔记21 小时前
Go语言临时对象池:sync.Pool的原理与使用
开发语言·后端·golang