C++ 函数详解

目录

函数概述

函数的分类

函数的参数

函数的调用

函数的嵌套调用

函数的链式访问

函数声明和定义

函数递归


函数概述

函数——具有某种功能的代码块

一个程序中我们经常会用到某种功能,如两数相加,如果每次都在需要用到时实现,那么就显得过于繁杂。此时,我们就可以将两数相加的功能封装起来,在需要使用的地方进行函数调用即可。

函数的分类

库函数:

  • 我们知道在我们学习C语言编程的时候,总是在一个代码编写完成之后迫不及待的想知道结果,想把这个结果打印到我们的屏幕上看看。这个时候我们会频繁的使用一个功能:将信息按照一定的格式打印到屏幕上(printf)。
  • 在编程的过程中我们会频繁的做一些字符串的拷贝工作(strcpy)。
  • 在编程是我们也计算,总是会计算n的k次方这样的运算(pow)。

像上面我们描述的基础功能,它们不是业务性的代码。我们在开发的过程中每个程序员都可能用的到,为了支持可移植性和提高程序的效率,所以C语言的基础库中提供了一系列类似的库函数,方便程序员进行软件开发。

那么库函数都有哪些呢?这里我给大家提供一个使用频率非常高的网站:
http://www.cplusplus.com(简易轻便,缺点是全英文但并不影响阅读)在里面我们可查询到各种各样的库函数及其使用方法。

库函数数量很多,无法一一列举,只能对其分类:

  • IO函数
  • 字符串操作函数
  • 字符操作函数
  • 内存操作函数
  • 时间/日期函数
  • 数学函数
  • 其他库函数 

自定义函数:

如果库函数能干所有的事情,那还要程序员干什么?

所以更加重要的是自定义函数

自定义函数和库函数一样,有函数名返回值类型函数参数

但是不一样的是这些都是我们自己来设计。这给程序员一个很大的发挥空间。

函数的构成:

函数返回值类型  函数名  (参数1,参数2,参数3......)
{//...函数体
}

函数的参数

C语言中函数的参数一般分为两种:

  • 实际参数(实参):

真实传给函数的参数,叫实参。

实参可以是:常量、变量、表达式、函数等。

无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。

  • 形式参数(形参):

形式参数是指函数名后括号中的变量,因为形式参数只有在函数被调用的过程中才实例化(分配内存单元),所以叫形式参数。形式参数当函数调用完成之后就自动销毁了。因此形式参数只在函数中有效。

举例:交换两个变量num1和num2中的值;

#include <stdio.h>
//实现成函数,但是不能完成任务
void Swap1(int x, int y) {int tmp = 0;tmp = x;x = y;y = tmp;
}
//正确的版本
void Swap2(int* px, int* py) {int tmp = 0;tmp = *px;*px = *py;*py = tmp;
}
int main()
{int num1 = 1;int num2 = 2;Swap1(num1, num2);printf("Swap1::num1 = %d num2 = %d\n", num1, num2);Swap2(&num1, &num2);printf("Swap2::num1 = %d num2 = %d\n", num1, num2);return 0;
}

运行结果如图所示,至于为什么两个函数的运行结果不相同,原因如下:

  • 首先 Swap1 和 Swap2 函数中的参数 x,y,px,py 都是形式参数。在main函数中传给 Swap1 的 num1 ,num2 和传给 Swap2 函数的 &num1 , &num2 是实际参数。
  • 在调用函数Swap1时,形参 x 和 y 拥有自己的空间,同时拥有了和实参一模一样的内容。此时x=1,y=2,当函数执行完之后,x=2,y=2。x和y确实交换了值,但与num1和num2并没有什么关系。
  • Swap2不同,形参 px、py是指针变量(int*为指针类型,后期会讲),对它俩进行解引用后再将值交换,实际上交换的就是num1和num2所在空间的内容。

所以我们可以简单的认为:形参实例化之后其实相当于实参的一份临时拷贝。
 

函数的调用

函数的调用分为两种:

传值调用;函数的形参和实参分别占有不同内存块,对形参的修改不会影响实参。(例如Swap1)

传址调用;传址调用是把函数外部创建变量的内存地址传递给函数参数的一种调用函数的方式。这种传参方式可以让函数和函数外边的变量建立起真正的联系,也就是函数内部可以直接操作函数外部的变量。(例如Swap2)

函数的嵌套调用

 在一个函数中可以调用另外一个函数

#include<stdio.h>
int add(int x, int y)//函数1
{return x + y;
}int addplus(int x, int y,int z)//函数2
{int sum = 0;sum = add(x, y);//调用函数1return sum + z;
}
int main()
{int num1 = 1;int num2 = 2;int num3 = 3;int pr=addplus(num1, num2, num3);printf("%d\n", pr);return 0;
}

函数的链式访问

把一个函数的返回值作为另外一个函数的参数

#include<stdio.h>
int add(int x, int y)//add函数返回值是int类型
{return x + y;
}#include <stdio.h>
int main()
{int num1 = 1;int num2 = 2;printf("%d\n", add(num1,num2));printf("%d", printf("%d", printf("%d", 43)));//结果是啥?//注:printf函数的返回值是打印在屏幕上字符的个数return 0;
}

函数声明和定义

函数声明:

  • 告诉编译器有一个函数叫什么,参数是什么,返回类型是什么。但是具体是不是存在,函 数声明决定不了;

  • 函数的声明一般出现在函数的使用之前。要满足先声明后使用

  • 函数的声明一般要放在头文件中的。

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>//申请一个结点
SLTNode* BuySLTNode(SLTDataType data);
//创建一个链表,包含数据为0~n
SLTNode* CreateSList(int n);
//释放内存
void SLTDestroy(SLTNode** pphead);
//尾插
void SLTPushBack(SLTNode** pphead, SLTDataType data);
//尾删
void SLTPopBack(SLTNode** pphead);
//...

 函数的定义是指函数的具体实现,交待函数的功能实现

SLTNode* BuySLTNode(SLTDataType data)
{SLTNode* newNode = (SLTNode*)malloc(sizeof(SLTNode));//检查是否申请成功if (newNode == NULL){perror("malloc fail");exit(-1);}//对newNode进行初始化newNode->data = data;newNode->next = NULL;//返回申请成功的结点return newNode;
}SLTNode* CreateSList(int n)
{...//过程省略
}void SLTDestroy(SLTNode** pphead)
{...//过程省略
}void SLTPushBack(SLTNode** pphead, SLTDataType data)
{...//过程省略
}
void SLTPopBack(SLTNode** pphead)
{...//过程省略
}

函数递归

 程序调用自身的编程技巧称为递归( recursion)

递归做为一种算法在程序设计语言中广泛应用。 一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解, 递归策略 只需少量的程序就可描述出解题过程所需要的多次重复计算,大大地减少了程序的代码量。

递归的主要思考方式在于:把大事化小。

函数递归的两个必要条件:

  • 存在限制条件,当满足这个限制条件的时候,递归便不再继续;

  • 每次递归调用之后越来越接近这个限制条件。

举例
接受一个整型值(无符号),按照顺序打印它的每一位(例如:输入:1234,输出 1 2 3 4)。

#include <stdio.h>
void print(int n) 
{if (n > 9){print(n / 10);}printf("%d ", n % 10);
}
int main()
{int num = 1234;print(num);return 0;
}

函数在被调用时会建立函数栈帧(简单理解为建立函数栈帧就是在内存中申请了一块中间来运作函数执行)。有些情况下,递归并不是特别高效,例如遇到斐波那契数列的问题时,就不再那么实用(虽然递归的实现方式易于理解代码)。

举例

//求斐波那契数列中第n个数
int fib(int n) {if (n <= 2)return 1;elsereturn fib(n - 1) + fib(n - 2);
}

这段代码理论上可以求出任何n的结果,但是实际上当n等于50左右程序就会挂掉。原因是这段代码的算法对内存的消耗巨大。

在调试 fib 函数的时候,如果你的参数比较大,那就会报错: stack overflow(栈溢出)这样的信息。系统分配给程序的栈空间是有限的,但是如果出现了死循环,或者(死递归),这样有可能导致一直开辟栈空间,最终产生栈空间耗尽的情况,这样的现象我们称为栈溢出

那如何解决上述的问题?

  • 将递归改写成非递归;
  • 使用static对象替代 nonstatic 局部对象。在递归函数设计中,可以使用 static对象替代nonstatic局部对象(即栈对象),这不 仅可以减少每次递归调用和返回时产生和释放 nonstatic 对象的开销,而且static 对象还可以保存递归调用的中间状态,并且可为各个调用层所访问;

举例:下面代码就采用了非递归的方式来实现:

//求第n个斐波那契数
int fib(int n) {int result;int pre_result;int next_older_result;result = pre_result = 1;while (n > 2){n -= 1;next_older_result = pre_result;pre_result = result;result = pre_result + next_older_result;}return result;
}
  • 许多问题是以递归的形式进行解释的,这只是因为它比非递归的形式更为清晰。

  • 但是这些问题的迭代实现往往比递归实现效率更高,虽然代码的可读性稍微差些

  • 当一个问题相当复杂,难以用迭代实现时,此时递归实现的简洁性便可以补偿它所带来的运行时开销。

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

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

相关文章

第4章 互联网

文章目录 4.1 计算机网络基础 94 4.1.1 计算机网络的基本概念 94 4.1.2 局域网的基本原理 96 4.1.3 局域网协议与应用 98 4.2 Internet 100 4.2.1 TCP/IP 101 4.2.2 TCP/IP应用 106 4.2.3 网络操作系统的功能 112 4.2.4 网络安全的概念 116 4.3 计算机软件编程基础 …

唯创知音WTR096A-16S语音芯片:引领喊话器声音还原新时代

随着科技的不断发展&#xff0c;声音在各种场景中扮演着至关重要的角色。在喊话器领域&#xff0c;唯创知音的WTR096A-16S语音芯片以其卓越的性能和声音还原度的提高&#xff0c;为喊话器应用带来了全新的体验。 1. 声音还原度的升级 WTR096A-16S语音芯片采用16位数模转换器&…

机器学习---线性回归算法

1、什么是回归&#xff1f; 从大量的函数结果和自变量反推回函数表达式的过程就是回归。线性回归是利用数理统计中回归分析来确定两种或两种以上变量间相互依赖的定量关系的一种统计分析方法。 2、一元线性回归 3、多元线性回归 如果回归分析中包括两个或两个以上的自变量&a…

VIM光标移动和翻页快捷键-包含vim帮助文档截图

光标移动到行首(行首没有空格)&#xff1a; ^ 光标移动到行首(行首有空格)&#xff1a; 数字0 光标移动到行尾&#xff1a; $ 移动到指定行&#xff1a;7G(数字加一个大G&#xff09; 光标移动到文件开始&#xff1a;gg(两个小g) 光标移动到文件末尾&#xff1a;G(一个大G&…

二叉树的实现(纯C语言版)

目录 1.实现的接口1.1通过前序遍历的数组"ABD##E#H##CF##G##"构建二叉树 1.2 二叉树销毁1.3二叉树节点个数1.4二叉树第k层节点个数1.5 二叉树查找值为x的节点1.6二叉树前序遍历1.7二叉树中序遍历1.8二叉树后序遍历1.9层序遍历1.10判断二叉树是否是完全二叉树1.11 二叉…

AI:大语言模型训练方法 - 机器学习

Transformer Transformer是一种深度学习的模型架构&#xff0c;特别适用于自然语言处理任务。Transformer 模型的核心创新在于其 "自注意力"&#xff08;Self-Attention&#xff09;机制&#xff0c;这种机制使得模型可以有效地捕捉输入数据中的长距离依赖关系。 T…

【python】pip install 国内源

背景 python很多库都需要用国内镜像才能比较快&#xff0c;常用的有临时换源或长久换源。 临时换源 pip install [库] -i [源]国内源常用的有以下几种。 阿里云 http://mirrors.aliyun.com/pypi/simple/ 豆瓣(douban) http://pypi.douban.com/simple/ 清华大学 https://pyp…

数据结构 | 查漏补缺之DFS、BFS、二次探测再散列法、完全二叉树、深度计算

目录 DFS&BFS 哈希表-二次探测再散列法 完全二叉树&深度计算 排序 快速排序-挖坑法 插入、选择、冒泡、区别 DFS&BFS 哈希表-二次探测再散列法 完全二叉树&深度计算 排序 快速排序-挖坑法 插入、选择、冒泡、区别 插入从第一个元素开始&#xff0c…

利用yolov5输出提示框,segment-anything生成掩膜实现图像的自动标注

文章目录 一. 创建环境二. 下载模型文件三. 编辑代码 一. 创建环境 anaconda下新建一个环境 conda create -n yolo-sam python3.8激活新建的环境 conda activate yolo-sam更换conda镜像源 conda config --add channels https://mirrors.tuna.tsinghua.edu.cn/anaconda/pkgs/fre…

el-input-number设置step、stepStrictly后,数据精度丢失的问题

el-input-number的配置 <el-input-numberv-else-if"colInputType(column, row) number"v-model"row[column.key]":placeholder"${$t(documentation.pleaseInput)}":controls"false":min"minFn(column, row)":max"…

企业CIO为何选择SD-WAN技术进行组网?

在这个信息技术快速发展的时代&#xff0c;网络已经成为企业的中枢神经系统&#xff0c;它不仅连接了分布在全球各地的办公室和员工&#xff0c;还提供了无缝的数据交流和协作环境。随着企业规模的扩大和数字化转型的加速&#xff0c;企业需要高效、安全、可靠的网络连接来应对…

C++如何进行异常安全性处理和资源管理?

在C中&#xff0c;异常安全性处理和资源管理是确保程序在面对异常情况时能够正确处理资源的重要方面。下面是一些常见的方法和技术&#xff0c;可以帮助实现异常安全性和有效地管理资源&#xff1a; RAII&#xff08;资源获取即初始化&#xff09;&#xff1a;RAII是一种C编程技…

vsftpd.confg 常用配置,Beyond Compare 测试可用

vsftpd.confg 常用配置,备份一下, 经常配置好久 , 以后直接粘贴即可. Beyond Compare 测试可用. # Example config file /etc/vsftpd.conf # # The default compiled in settings are fairly paranoid. This sample file # loosens things up a bit, to make the ftp daemon m…

注册阿里云,免费领云服务器

注册阿里云&#xff0c;免费领云服务器&#xff0c;每月280元额度&#xff0c;3个月试用时长&#xff0c;可快速搭建网站/小程序&#xff0c;部署开发环境&#xff0c;开发多种企业应用&#xff0c;共3步骤即可免费领取阿里云服务器&#xff0c;阿里云服务器网aliyunfuwuqi.com…

Python与PHP:编写大型爬虫的适用性比较

目录 一、引言 二、Python编写爬虫的优势 1、强大的数据处理能力 2、丰富的网络库和框架 3、良好的可读性和易维护性 4、社区支持和生态系统 三、PHP编写爬虫的优势 1、简单易学 2、广泛的应用领域 3、高效的性能 4、灵活的请求处理方式 四、大型爬虫的编写实例&am…

Tauri魔法指南:从零开始开发一个桌面应用

摘要&#xff1a; 本文将以轻松幽默的笔调&#xff0c;带领读者探索如何使用Tauri这个前端魔法工具&#xff0c;从零开始开发一个跨平台的桌面应用。通过深刻的洞察和实际示例&#xff0c;让你轻松进入这个神奇的桌面开发领域。 引言 曾几何时&#xff0c;前端开发者们只是在…

springboot(ssmBBS论坛系统 论坛交流平台Java(codeLW)

springboot(ssmBBS论坛系统 论坛交流平台Java(code&LW) 开发语言&#xff1a;Java 框架&#xff1a;ssm/springboot vue JDK版本&#xff1a;JDK1.8&#xff08;或11&#xff09; 服务器&#xff1a;tomcat 数据库&#xff1a;mysql 5.7&#xff08;或8.0&#xff09;…

re:invent 2023 Amazon Q 初体验

授权声明&#xff1a;本篇文章授权活动官方亚马逊云科技文章转发、改写权&#xff0c;包括不限于在 Developer Centre&#xff0c;知乎&#xff0c;自媒体平台&#xff0c;第三方开发者媒体等亚马逊云科技官方渠道 前言 亚马逊云科技在2023 re:Invent全球大会上宣布推出 Amazon…

计算机网络——数据链路层-差错检测(奇偶校验、循环冗余校验CRC)

目录 奇偶校验 循环冗余校验CRC 发送方操作 接收方操作 生成多项式 举例-1 举例-2 我们知道&#xff0c; 实际的通信链路都不是理想的&#xff0c;比特在传输过程中可能会产生差错&#xff1b;1可能变成0&#xff0c;而0也可能变成1&#xff0c;这称为比特差错。 如下…

[wordpiece]论文分析:Google’s Neural Machine Translation System

文章目录 一、论文解读1.1 模型介绍1.2 模型架构1.3 wordpiece 二、整体总结 论文&#xff1a;Google’s Neural Machine Translation System: Bridging the Gap between Human and Machine Translation 作者&#xff1a;Yonghui Wu, Mike Schuster, Zhifeng Chen, Quoc V. Le,…