C语言 | 函数核心机制深度解构:从底层架构到工程化实践

个人主页-爱因斯晨

文章专栏-C语言

在这里插入图片描述

引言

最近偷懒了,迷上了三国和李贺。给大家分享一下最喜欢的一句诗:吾不识青天高黄地厚,唯见月寒日暖来煎人寿。我还不是很理解27岁的李贺,如何写出如此绝笔。

正文开始,今天我们来探讨一下关于C语言中的函数部分

一、函数的概念:代码的 “模块化” 基石

1.1 函数的定义与意义

  • 定义:函数是一段可重复使用的代码块,具有输入(参数)处理逻辑(函数体)**和**输出(返回值)
  • 意义:
    • 复用性:避免重复编写相同逻辑(如多次计算最大值,只需调用 max 函数)。
    • 可读性:通过函数名(如 sortArray)直观理解功能,降低代码复杂度。
    • 可维护性:修改函数内部逻辑时,只需更新一处,不影响其他调用处。

1.2 函数的基本结构

返回类型 函数名(参数列表) {// 函数体:实现具体功能return 返回值; // 非void类型必须返回对应类型的值
}
  • 示例:计算两数之和

    int add(int a, int b) { // 返回int,参数a、b为intreturn a + b; // 返回和
    }
    

二、库函数:“开箱即用” 的工具集

关于库函数和其他语言中封装的函数在上篇文章中已经讲到了,详情请看[从库函数到API接口,深挖不同语言背后的“封装”与“调用”思想-CSDN博客]()

2.1 库函数的分类与头文件

  • 标准库:C 语言内置的函数集合,分为:

    • 输入输出(stdio.hprintf(输出)、scanf(输入)。
    • 字符串处理(string.hstrlen(字符串长度)、strcpy(字符串复制)。
    • 数学运算(math.hsqrt(开平方)、pow(幂运算)。
    • 内存管理(stdlib.hmalloc(动态内存分配)、free(释放内存)。
  • 头文件:包含库函数的声明

    (告诉编译器函数的存在、参数和返回值)。使用库函数前必须包含对应头文件,例如:

    #include <stdio.h> // 包含printf的声明
    int main() {printf("Hello, World!"); // 调用库函数return 0;
    }
    

2.2 库函数的使用步骤(以 fgets 为例)

  1. 查阅文档fgets 从文件中读取字符串,原型为 char *fgets(char *s, int size, FILE *stream);

  2. 包含头文件#include <stdio.h>fgets 声明在此头文件中)。

  3. 调用函数

    #include <stdio.h>
    int main() {char str[100];fgets(str, 100, stdin); // 从标准输入(键盘)读取最多99个字符(含'\0')printf("输入内容:%s", str);return 0;
    }
    
  4. 注意事项:

    • size 参数需小于数组长度(避免缓冲区溢出)。
    • 返回值为 NULL 表示读取失败(如文件结束)。

三、自定义函数:“按需定制” 的代码块

3.1 函数定义的详细语法

  • 返回类型:
    • void:无返回值(如仅打印信息的函数)。
    • 基本类型(intfloat 等):返回对应类型的值。
  • 参数列表:
    • 无参数void func()func()(C99 后允许省略 void)。
    • 有参数int add(int a, int b)ab 为形参,接收实参的值)。
  • 函数体:包含实现逻辑的代码,可使用 return 提前结束函数(void 函数用 return;)。

3.2 示例:实现 “判断素数” 函数

#include <stdio.h>
#include <math.h>// 自定义函数:判断n是否为素数(返回1是,0否)
int isPrime(int n) {if (n <= 1) return 0; // 1及以下不是素数for (int i=2; i<=sqrt(n); i++) { // 优化:只需检查到平方根if (n % i == 0) return 0; // 能整除,不是素数}return 1; // 是素数
}int main() {int num;printf("输入一个整数:");scanf("%d", &num);if (isPrime(num)) {printf("%d是素数\n", num);} else {printf("%d不是素数\n", num);}return 0;
}
  • 解释:
    • 形参 n 接收实参(用户输入的 num)。
    • 通过循环判断是否有因数,提前返回结果(提高效率)。

四、形参和实参:“值的传递与拷贝”

4.1 实参(实际参数)

  • 定义:调用函数时传递的具体值或变量(如 isPrime(num) 中的 num)。
  • 特点:
    • 可以是常量isPrime(7))、变量isPrime(num))、表达式isPrime(2+3))。
    • 传递方式:值传递(形参是实参的拷贝,修改形参不影响实参,除非传递地址)。

4.2 形参(形式参数)

  • 定义:函数定义时占位的参数(如 isPrime(int n) 中的 n)。
  • 特点:
    • 函数调用时分配内存,调用结束后释放(形参是临时变量)。
    • 值传递本质:形参是实参的副本(如 nnum 的拷贝,修改 n 不影响 num)。

4.3 地址传递(突破值传递限制)

// 交换两数(通过地址传递,修改实参)
void swap(int *a, int *b) {int temp = *a;*a = *b;*b = temp;
}int main() {int x=10, y=20;swap(&x, &y); // 实参是x、y的地址(传递指针)printf("x=%d, y=%d\n", x, y); // 输出x=20, y=10(实参被修改)return 0;
}
  • 解释:
    • 形参 *a*b 接收实参的地址(&x&y),通过解引用(*a)直接修改原变量的值。
    • 这是 值传递的特殊情况(传递地址,实现 “引用传递” 效果)。

五、return 语句:“函数的出口与结果”

5.1 return 的两种用法

  • 返回值:给调用者一个结果(如 return a + b; 返回和)。
  • 结束函数:提前退出函数(如 void 函数中的 return;,跳过后续代码)。

5.2 规则与示例

  • void 函数

    void printMessage() {printf("Hello!\n");return; // 可省略(函数体结束自动返回)
    }
    
  • 非 void 函数

    int max(int a, int b) {if (a > b) return a; // 返回a,结束函数return b; // 必有一个执行(确保返回值)
    }
    
  • 错误处理

    int divide(int a, int b) {if (b == 0) {printf("除数不能为0!\n");return -1; // 错误码(调用者根据返回值判断是否出错)}return a / b;
    }
    

六、数组作为函数参数:“传递指针与内存”

6.1 数组传参的本质

  • 数组名作为参数时,传递的是首元素的地址(即指针),函数内对数组的修改会影响原数组(因为操作同一块内存)。

  • 语法

    void printArray(int arr[], int size) { // 等价于int *arrfor (int i=0; i<size; i++) {printf("%d ", arr[i]); // 等价于*(arr+i)}
    }
    

6.2 示例:数组排序(冒泡排序)

#include <stdio.h>void bubbleSort(int arr[], int size) {for (int i=0; i<size-1; i++) {for (int j=0; j<size-i-1; j++) {if (arr[j] > arr[j+1]) { // 交换相邻元素int temp = arr[j];arr[j] = arr[j+1];arr[j+1] = temp;}}}
}int main() {int nums[] = {5, 3, 8, 1, 2};int size = sizeof(nums) / sizeof(nums[0]); // 计算数组长度bubbleSort(nums, size); // 传递数组名(首地址)和长度for (int i=0; i<size; i++) {printf("%d ", nums[i]); // 输出1 2 3 5 8(原数组已排序)}return 0;
}
  • 注意:函数无法自动获取数组长度(需手动传递 size),因为形参 arr 是指针(丢失长度信息)。

七、函数调用:嵌套与链式

7.1 嵌套调用(函数内调用其他函数)

void printHeader() {printf("===== 欢迎使用系统 =====\n");
}void printMenu() {printHeader(); // 嵌套调用printHeaderprintf("1. 登录\n2. 注册\n3. 退出\n");
}int main() {printMenu(); // 输出:===== 欢迎使用系统 ===== → 菜单选项return 0;
}

7.2 链式访问(函数返回值作为参数)

int add(int a, int b) { return a + b; }
int mul(int a, int b) { return a * b; }int main() {// 先算add(2,3)=5,再算mul(5,4)=20(链式调用)int result = mul(add(2, 3), 4); printf("结果:%d\n", result); // 输出20return 0;
}

八、函数的声明与定义:“多文件开发”

8.1 单个文件中的声明

  • 定义在前:直接调用(无需声明)。

  • 定义在后:需先声明(告诉编译器函数存在)。

    int add(int, int); // 声明(参数名可省略,只写类型)
    int main() {int res = add(3,5); // 调用时,编译器通过声明知道add存在return 0;
    }
    int add(int a, int b) { return a + b; } // 定义在后
    

8.2 多文件开发(模块化)

  • 步骤:

    1. 创建头文件(func.h):声明函数

      #ifndef FUNC_H
      #define FUNC_H
      int add(int a, int b); // 声明
      #endif
      
    2. 创建源文件(func.c):定义函数

      #include "func.h" // 包含头文件(双引号表示当前目录)
      int add(int a, int b) { return a + b; } // 定义
      
    3. 主文件(main.c):调用函数

      #include <stdio.h>
      #include "func.h" // 包含头文件,获取声明
      int main() {printf("%d\n", add(3,5)); // 调用func.c中的addreturn 0;
      }
      
  • 编译:需同时编译 main.cfunc.c(如 gcc main.c func.c -o main)。

8.3 static 关键字:“限制作用域”

  • 静态函数(static 修饰函数)

    • 作用:仅当前文件可见(其他文件无法调用,避免命名冲突)。

    • 示例(func.c

      static int add(int a, int b) { return a + b; } // main.c调用会报错(未定义)
      
  • 静态变量

    • 静态局部变量(函数内)

      生命周期为程序运行期(保留值,如计数器)。

      void count() {static int num = 0; // 第一次调用初始化,后续保留值num++;printf("第%d次调用\n", num);
      }
      // 调用:count() → 第1次,count() → 第2次(num保留1)
      
    • 静态全局变量(文件内,函数外):仅当前文件可见(同文件内函数可访问,其他文件不可见)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/88359.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/88359.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

uniapp真机调试“没有检测到设备,请插入设备或启动模拟器后点击刷新再试”

当真机调试&#xff0c;运行到安卓 APP基座 时&#xff0c;有时会检测不到设备&#xff0c;显示下面的问题&#xff1a;此时&#xff0c;可以通过下面的几种方法进行排查&#xff1a;1.在手机中找到“开发者选项”选项&#xff08;可在设置中搜索&#xff0c;如搜索不到&#x…

使用langchain连接llama.cpp部署的本地deepseek大模型开发简单的LLM应用

langchain是一个基于python实现的开源LLM开发框架&#xff0c;llama.cpp是一个基于C框架可以在本地部署大模型并开放服务端接口开放给外部应用使用。 本文结合langchain和llama.cpp&#xff0c;在本地部署轻量级的deepseek大模型&#xff0c;并构建一个简单的链式LLM应用&…

Serverless 数据库来了?无服务器数据库 vs 传统数据库有何不同?

随着云计算技术的迅猛发展&#xff0c;无服务器&#xff08;Serverless&#xff09;架构逐渐成为一种主流趋势。其中&#xff0c;Serverless 数据库作为云原生应用的重要组成部分&#xff0c;为开发者提供了前所未有的灵活性和成本效益。相比传统的数据库管理方式&#xff0c;S…

【读书笔记】如何画好架构图:架构思维的三大底层逻辑

【读书笔记】如何画好架构图&#xff1a;架构思维的三大底层逻辑 架构图并非技术人的“画功比拼”&#xff0c;而是一个团队、一个系统、一次项目从混沌走向清晰的关键抓手。它是系统的视觉语言&#xff0c;是让技术人员、产品经理、运营甚至老板都能站在统一上下文下讨论的“…

Maven 编译过程中发生了 Java Heap Space 内存溢出(OutOfMemoryError)

这个是我最近遇到的&#xff0c;因为本人最近换了电脑&#xff0c;这个电脑的前任是配置好了环境&#xff0c;但是当我用这个环境去做另外一个项目的时候&#xff0c;在maven构建war和jar包的时候&#xff0c;报了这个内存溢出mvn clean install 就给我报错了[ERROR] Failed to…

C++ 模板参数展开

C 模板参数展开一、获取可变参数大小二、通过模版循环继承的方式来展开可变参数三、改用Using去实现循环继承一、获取可变参数大小 背景&#xff1a; FLen<int, char, long> Len; 我想要获取模板参数类型的总大小 template<typename T,typename ...ParamTypes> c…

零基础入门物联网-远程门禁开关:云平台创建

一、 onenet云平台注册创建 远程开关的信息传输依赖云平台&#xff0c;本教程以 OneNET - 中国移动物联网开放平台为例进行操作&#xff0c;具体步骤如下&#xff1a; 1、平台账号创建 点击 OneNET - 中国移动物联网开放平台进入官网 点击页面中的 “登录” 按钮&#xff0c;…

html页面,当鼠标移开A字标就隐藏颜色框

html页面代码&#xff1a;<!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><title>颜色选择器</title><style>body {font-family: "Microsoft YaHei", sans-serif;padding: 20px;}.c…

保姆级搭建harbor私有仓库与docker-ce教程与使用教程

搭建harbor仓库[rootharbor ~]# vim cat /etc/host192.168.121.12 harbor[rootharbor ~]# vim /etc/hostnameharbor导入 harbor 项目镜像[rootharbor ~]# tar -zxf harbor-v2.9.2.tgz -C /usr/local/[rootharbor ~]# cd /usr/local/harbor[rootharbor harbor]# docker load -i…

【Linux】Rocky Linux 安装 Docker 与 Docker-Compose

Docker 安装步骤 1. 安装必要的软件包 sudo yum install -y yum-utils sudo yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo2. 安装Docker sudo yum install docker-ce docker-ce-cli containerd.io如果出现 SSL 证书错误&#xf…

揭示独特模式:Elasticsearch 中 significant terms 聚合指南

作者&#xff1a;来自 Elastic Alexander Dvila 了解如何使用 significant terms 聚合来发现你数据中的洞察。 更多阅读&#xff1a;Elasticsearch&#xff1a;significant terms aggregation Elasticsearch 拥有大量新功能&#xff0c;可以帮助你为你的使用场景构建最佳搜索解…

pandas.DataFrame中axis参数

明确axis0与axis1的区别和联系&#xff0c; 假设有一个 DataFrame&#xff1a;indexAB012134axis0&#xff08;沿行方向&#xff09;&#xff1a; 操作会垂直向下进行&#xff0c;对每一列单独处理。 例如&#xff1a;df.sum(axis0) 会对列 A 和列 B 分别求和&#xff0c;结果是…

深度学习 最简单的神经网络 线性回归网络

用最简单的线性模型讲清 神经网络 训练全流程,让你 5 分钟看懂AI 是怎么学会预测的 🔥 1 真实神经元结构 📊 真实神经元包括: 树突 接收其他神经元传来的电信号(输入)。 细胞核 负责整合输入信号并产生动作电位。 轴突 传导动作电位到下一个神经元。 突触 释放神经递质…

k8s Mutating Admission Webhook 实现超卖

目录 1.什么是 Mutating Admission Webhook&#xff1f; 2.如何用 Mutating Admission Webhook 实现超卖&#xff1f; 3.实现超卖 3.1 理解目标 3.2 前置准备 3.3 开发 Mutating Webhook 3.4 配置 Webhook Server TLS 认证 3.5 注册 MutatingWebhookConfiguration 3.6…

为 Go-llm-cpp 接入 Web API 接口,创建 Chatbot 聊天机器人

接续上一篇&#xff0c;用 Go 打造本地 LLM 聊天机器人&#xff1a;整合 llm-go 与 go-llama.cpp&#xff0c;此篇开始建构前端与 API 接口 执行环境需求 • ✅ Go 1.20 • ✅ C toolchain&#xff08;macOS: Xcode Command Line Tools / Linux: g&#xff09; • ✅ GGUF 格式…

Docker笔记-Docker Compose

Docker笔记-Docker Compose Compose 是用于定义和运行多容器 Docker 应用程序的工具&#xff0c;通过 Compose 您可以使用 YML 文件来配置应用 程序需要的所有服务。然后&#xff0c;使用一个命令&#xff0c;就可以从 YML 文件配置中创建并启动所有服务。 Compose 使用的三个步…

n1 armbian 安装桌面环境并启用xrdp远程登录

armbian-config armbian-software201frpcrootarmbian:~# armbian-software [ STEPS ] Start selecting software [ Current system: ubuntu/noble ]... ──────────────────────────────────────────────────────────…

从传统到智能:地质灾害风险评估、易发性分析与灾后重建;AI大语言模型DeepSeek、ChatGPT、GIS、Python和机器学习深度融合

地质灾害是指全球地壳自然地质演化过程中&#xff0c;由于地球内动力、外动力或者人为地质动力作用下导致的自然地质和人类的自然灾害突发事件。在降水、地震等自然诱因的作用下&#xff0c;地质灾害在全球范围内频繁发生。我国不仅常见滑坡灾害&#xff0c;还包括崩塌、泥石流…

便捷的电脑自动关机辅助工具

软件介绍 本文介绍的软件是一款电脑上实用的倒计时和关机助手。 软件特性 这款关机助手十分贴心&#xff0c;它是一款无需安装的小软件&#xff0c;体积仅60KB&#xff0c;不用担心占用电脑空间&#xff0c;打开即可直接使用。 操作方法 你只需设置好对应的关机时间&#x…

Fiddler-关于抓取Android手机包,安装证书后页面加载失败,提示当前证书不可信存在安全风险的问题

Fiddler-关于抓取Android手机包&#xff0c;安装证书后页面加载失败&#xff0c;提示当前证书不可信存在安全风险的问题Fiddler-关于抓取Android手机包&#xff0c;安装证书后页面加载失败&#xff0c;提示当前证书不可信存在安全风险的问题原因解决方法Fiddler-关于抓取Androi…