C++自定义函数详解

news2025/1/17 2:58:01

个人主页:PingdiGuo_guo

收录专栏:C++干货专栏

铁汁们新年好呀,今天我们来了解自定义函数。

文章目录

1.数学中的函数

2.什么是自定义函数

3.自定义函数如何使用?

4.值传递和引用传递(形参和实参区分)

1.值传递

2. 引用传递

3. 形参与实参的区别

5.自定义递归函数

6.嵌套调用和链式调用

1.链式调用

2.嵌套调用

7.自定义函数和库函数的对比

8.自定义函数的练习

9.总结


1.数学中的函数

在数学中,函数是一种基本的数学对象,它建立了一个集合(称为定义域)中的每个元素与另一个集合(称为值域)中的唯一元素之间的对应关系。简单地说,函数就是一个规则,它接收一个或多个输入值(也称为自变量),并据此产生一个确定的输出值(称为函数值或因变量)。

数学上的函数通常用 ( f ) 表示,并写作 ( y = f(x) ),这里的 ( x ) 是自变量,( y ) 是相应于 ( x ) 的函数值。例如,线性函数 ( f(x) = mx + b ) 就是一条直线的方程,其中 ( m ) 是斜率,( b ) 是截域。

2.什么是自定义函数

自定义函数是在C++或其他编程语言中由程序员自行创建的函数,用于封装一组操作或计算逻辑,以便在多个地方重复使用或者模块化代码。自定义函数允许开发者按照自己的需求来定制功能,并且可以根据程序的具体情况调整其行为。
 

3.自定义函数如何使用?

1. 声明函数原型:
在函数体之前(通常在头文件或源文件的开始部分),你需要声明函数,包括返回类型、函数名以及参数列表(如果有的话):

 

// 声明一个自定义函数原型
double calculateArea(double radius);



这个例子声明了一个名为calculateArea的函数,它接受一个double类型的参数(半径),并返回一个double类型的值(面积)。

2. 定义函数体:
在函数声明之后,你需要定义函数的实际实现,即写出完成特定任务的代码块:

 

// 定义函数体
double calculateArea(double radius) {
double area = 3.14159 * radius * radius;
return area;
}



3. 调用函数:
在程序的适当位置,你可以像使用库函数一样调用自定义函数:


 

int main() {
double myRadius = 5.0;
double result = calculateArea(myRadius);
cout << "The area of the circle with radius " << myRadius << " is: " << result << endl;
return 0;
}

4.值传递和引用传递(形参和实参区分)

1.值传递

- 当函数通过值传递方式接收参数时,它会创建实参的副本(复制一份实参的值)并将这个副本传递给函数内部的形参。
- 形参的变化不会影响实参,因为两者存储在不同的内存位置。
- 值传递通常适用于不需要修改实参值的情况。

示例:

void swapByValue(int a, int b) {
int temp = a;
a = b;
b = temp;
}
int main() {
int x = 10, y = 20;
swapByValue(x, y); // 调用函数后,x和y的值不变,因为交换的是函数内部形参的值
}




2. 引用传递


- 引用传递是将实参的引用(也就是实参的别名)传递给函数的形参,这样形参就直接指向实参的内存地址。
- 如果在函数内部改变了引用类型的形参,那么相应地,实参的值也会被修改。
- 引用传递常用于需要更改实参或避免大型数据结构复制的成本时。

示例:
 

void swapByReference(int& a, int& b) {
int temp = a;
a = b;
b = temp;
}
int main() {
int x = 10, y = 20;
swapByReference(x, y); // 调用函数后,x和y的值会互换,因为函数内部修改的是实参的内存
}




3. 形参与实参的区别


- 实参(Actual Parameter):在函数调用过程中,传递给函数的实际值。例如上面例子中的 x和 y 就是实参。

- 形参(Formal Parameter):在函数定义中,用来接收传递进来的值的变量。如上述例子中的 a 和 b 是形参,它们在函数内部代表了从外部传入的值。

总结来说,在函数调用过程中,实参是实际提供的值,而形参是函数声明中用来接收这些值的地方。在C++中,值传递和引用传递决定了函数操作的是实参的副本还是实参本身。

5.自定义递归函数

递归调用函数是在函数内部调用自身的一种编程技术。这种技术在解决具有自我重复性质的问题时非常有用,如树的遍历、动态规划中的某些子问题解法、分治算法中的问题拆解等。

递归函数通常包含两个基本部分:
- 基本情况(Base Case):这是递归结束的条件,即当达到某个特定条件时,不再进行下一层递归调用,而是直接返回一个确定的结果。
- 递归步骤(Recursive Step):在此阶段,函数通过调用自身来解决规模更小的相同问题,直到满足基本情况为止。

下面是一个简单的递归函数示例,计算阶乘(Factorial):

 

#include <iostream>

// 递归函数实现阶乘
int factorial(int n) {
// 基本情况:0的阶乘为1
if (n == 0)
return 1;
// 递归步骤:n的阶乘等于n乘以(n-1)的阶乘
else
return n * factorial(n - 1);
}

int main() {
int num = 5;
std::cout << "The factorial of " << num << " is: " << factorial(num) << std::endl;
return 0;
}


在这个例子中,factorial 函数在每次递归调用时都会将问题规模缩小(减少参数n的值),直到遇到基本情况n为0时停止递归,然后逐层返回计算结果。

6.嵌套调用和链式调用

1.链式调用

链式调用是一种编程风格,可以在调用多个方法时使用点操作符将它们连接在一起,形成一个链。在每个方法调用后,都会返回一个对象,可以继续调用其他方法。这样就可以在一行代码中实现多个方法的调用,使代码更加简洁和可读性更高。

步骤一:定义一个支持链式访问的类

 

class Chainable {
public:
// 构造函数
Chainable(int initial_value) : value(initial_value) {}

// 第一个链式方法(假设增加值)
Chainable& increment(int step = 1) {
value += step;
return *this; // 返回当前对象的引用以支持链式调用
}

// 第二个链式方法(假设加倍)
Chainable& doubleValue() {
value *= 2;
return *this;
}
private:
int value;
};



步骤二:创建对象并开始链式调用

 

int main() {
// 创建Chainable对象
Chainable obj(5);

// 链式访问与调用
obj.increment(3).doubleValue();

// 这里原本可能还有其他方法继续调用...
// ...

// 最终可以验证对象的值
std::cout << "Final value: " << obj.value << std::endl;

return 0;
}



在上述代码中,首先定义了一个名为Chainable的类,它有两个链式方法:increment()和doubleValue()。这两个方法都在执行相应的操作之后返回了对当前对象的引用,使得可以连续调用这些方法,形成链式调用。在main()函数中展示了如何使用这种方式连续调用两个方法来处理同一个对象。

2.嵌套调用

嵌套调用是指在一个方法内部调用另一个方法,并且被调用的方法又可以继续调用其他方法,形成多层嵌套的调用关系。嵌套调用可以在程序中实现复杂的逻辑和功能,将代码进行模块化和组合,提高代码的可复用性和可维护性。

在嵌套调用中,每个方法的返回值可以作为下一个方法的参数,这样可以将多个方法串联起来,实现一系列的操作。嵌套调用的层级可以根据具体需求进行设计,可以是两层、三层,甚至更多。

步骤1:定义一个包含内部函数的外部函数:
 

void outerFunction(int x) {
// ...
void innerFunction();
// ...
}


步骤2:定义内部函数的具体实现:
 

void outerFunction(int x) {
std::cout << "In outer function with x = " << x << std::endl;

void innerFunction() {
std::cout << "In inner function." << std::endl;
}
}


步骤3:在外部函数内部调用内部函数:
 

void outerFunction(int x) {
// ...
innerFunction(); // 这里是嵌套调用发生的地方
// ...
}


步骤4:完整外部函数及其调用示例:

void outerFunction(int x) {
std::cout << "In outer function with x = " << x << std::endl;

void innerFunction() {
std::cout << "In inner function." << std::endl;
}

innerFunction();

std::cout << "Continuing in outer function after inner call." << std::endl;
}

int main() {
outerFunction(5); // 在主函数中调用outerFunction
return 0;
}



输出:
 

In outer function with x = 5
In inner function.
Continuing in outer function after inner call.

7.自定义函数和库函数的对比

- 自定义函数:
- 由程序员根据具体需求编写。
- 可以自由命名,具有高度灵活性。
- 具体实现完全取决于程序员,可修改、优化或扩展。
- 只能在包含它们的源文件或链接到它们的库中调用。

- 库函数:
- 预先编写好的、经过充分测试的标准函数,通常由语言本身或第三方库提供。
- 名称和功能通常是标准化的,如C++标准库中的std::sort()、std::cos()等。
- 不需要程序员重新实现,只需包含适当的头文件并在程序中正确调用即可。
- 通常跨平台兼容性更好,因为它们遵循语言规范和标准。
- 提供的功能广泛多样,涵盖基础数据结构操作、数学计算、输入/输出处理等多个方面。

总结来说,自定义函数的核心价值在于它可以针对性地解决特定的问题,并且有助于提高代码的复用性和组织性;而库函数则是为了提供通用的、高效的功能组件,减少重复劳动,提升开发效率和程序质量。

8.自定义函数的练习

假设我们要编写一个C++函数,它接受两个整数作为参数,返回它们的最大公约数(Greatest Common Divisor, GCD)。这个功能可以通过使用欧几里得算法(Euclidean algorithm)来实现。下面是这个函数的定义和使用示例:

 

#include <iostream>

// 自定义函数:计算两个整数的最大公约数
int gcd(int a, int b) {
// 边界条件:如果b为0,则a就是最大公约数
if (b == 0)
return a;

// 否则,递归地计算(a mod b)和b的最大公约数
return gcd(b, a % b);
}

int main() {
int num1 ,num2 ;
std::cin>>num1>>num2;

// 调用自定义函数并输出结果
std::cout << "The Greatest Common Divisor (GCD) of " << num1 << " and " << num2 << " is: " << gcd(num1, num2) << std::endl;

return 0;
}



上述代码首先定义了一个名为gcd的函数,该函数采用两个整数作为参数,并使用递归方法找到它们的最大公约数。在main函数中,我们创建了两个变量并调用了gcd函数来计算它们的最大公约数,最后将结果输出到控制台。

9.总结

本篇博客到这里就结束了,感谢大家的支持与观看,如果有好的建议欢迎留言,谢谢大家啦!

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

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

相关文章

ES实战-book笔记1

#索引一个文档,-XPUT手动创建索引, curl -XPUT localhost:9200/get-together/_doc/1?pretty -H Content-Type: application/json -d {"name": "Elasticsearch Denver","organizer": "Lee" } #返回结果 {"_index" : "g…

极狐GitLab 与钉钉的集成实践

DingTalk OAuth 2.0 OmniAuth provider * 引入于 14.5 版本。 您可以使用您的钉钉账号登录极狐GitLab。 登录钉钉开放平台&#xff0c;创建应用。钉钉会生成一个客户端 ID 和密钥供您使用。 登录钉钉开放平台。 在顶部栏上&#xff0c;选择 应用程序开发 > 企业内部开发&am…

修改SpringBoot中默认依赖版本

例如SpringBoot2.7.2中ElasticSearch版本是7.17.4 我希望把它变成7.6.1

物联网数据隐私保护技术

在物联网&#xff08;IoT&#xff09;的世界中&#xff0c;无数的设备通过互联网连接在一起&#xff0c;不断地收集、传输和处理数据。这些数据有助于提高生产效率、优化用户体验并创造新的服务模式。然而&#xff0c;随着数据量的剧增&#xff0c;数据隐私保护成为了一个不能忽…

CSP-202012-1-期末预测之安全指数

CSP-202012-1-期末预测之安全指数 题目很简单&#xff0c;直接上代码 #include <iostream> using namespace std; int main() {int n, sum 0;cin >> n;for (int i 0; i < n; i){int w, score;cin >> w >> score;sum w * score;}if (sum > 0…

力扣刷题之旅:进阶篇(三)

力扣&#xff08;LeetCode&#xff09;是一个在线编程平台&#xff0c;主要用于帮助程序员提升算法和数据结构方面的能力。以下是一些力扣上的入门题目&#xff0c;以及它们的解题代码。 --点击进入刷题地址 一、动态规划&#xff08;DP&#xff09; 首先&#xff0c;让我们来…

Linux中ps/kill/execl的使用

ps命令&#xff1a; ps -aus或者ps -ajx或者 ps -ef可以查看有哪些进程。加上 | grep "xxx" 可以查看名为”xxx"的进程。 ps -aus | grep "xxx" kill命令&#xff1a; kill -9 pid 杀死某个进程 kill -l 查看系统有哪些信号 execl函数&#…

C#上位机与三菱PLC的通信05--MC协议之QnA-3E报文解析

1、MC协议回顾 MC是公开协议 &#xff0c;所有报文格式都是有标准 &#xff0c;MC协议可以在串口通信&#xff0c;也可以在以太网通信 串口&#xff1a;1C、2C、3C、4C 网口&#xff1a;4E、3E、1E A-1E是三菱PLC通信协议中最早的一种&#xff0c;它是一种基于二进制通信协…

再识C语言 DAY17 【什么是原码、反码和补码】

文章目录 前言本文总结于此文章 一、知识补充二、原码三、反码四&#xff0c;补码 总结如果您发现文章有错误请与我留言&#xff0c;感谢 前言 本文总结于此文章 一、知识补充 通常&#xff0c;1字节包含8位。C语言用字节&#xff08;byte&#xff09;表示储存系统字符集所需…

【原理图PCB专题】Cadence17.4 PCB位号重排与反标

在文章:【原理图专题】Cadence 16.6如何把PCB元件位号重排并反标到原理图 中我们讲到了Cadence16.6版本对原理图进行反标的操作。 对于反标之前我们是通过如下所示的绘制流程来讲的,一般在首板或是大改板操作器件里有很多不同的很大的位号,这时我们可以通过Backannotate功能…

kettle--文本文件输出有空格解决方案

在kettle文本文件输出时&#xff0c;不管如何设置字段类型和长度&#xff0c;导出的数据都会有空格&#xff0c;遇到这一问题&#xff0c; 可以在文本文件输出控件中勾选这一项&#xff0c;即可解决这一问题。 文本文件输出&#xff1a;

(每日持续更新)jdk api之ObjectInputStream基础、应用、实战

博主18年的互联网软件开发经验&#xff0c;从一名程序员小白逐步成为了一名架构师&#xff0c;我想通过平台将经验分享给大家&#xff0c;因此博主每天会在各个大牛网站点赞量超高的博客等寻找该技术栈的资料结合自己的经验&#xff0c;晚上进行用心精简、整理、总结、定稿&…

单片机学习路线(简单介绍)

学习单片机对于电子爱好者和未来的嵌入式系统工程师来说是一段激动人心的旅程。单片机因其强大的功能、灵活性以及在各种智能设备中的广泛应用&#xff0c;成为了电子和计算机科学领域一个不可或缺的组成部分。如果你对如何开始这段旅程感到好奇&#xff0c;那么你来对地方了。…

kmeans聚类选择最优K值python实现

Kmeans算法中K值的确定是很重要的。 下面利用python中sklearn模块进行数据聚类的K值选择 数据集自制数据集&#xff0c;格式如下&#xff1a; 维度为3。 ①手肘法 手肘法的核心指标是SSE(sum of the squared errors&#xff0c;误差平方和)&#xff0c; 其中&#xff0c;Ci是第…

编译器选择:VSCode安装MarkDown插件

目录 1.打开vscode2.点击扩展选项3.搜索 Markdown Preview Enhanced插件4.使用test示例5.然后右键鼠标选择打开侧边预览6.实时预览效果 注&#xff1a;本篇文章默认用户安装了vscode&#xff0c;未安装的可以自行查找教程安装。 注&#xff1a;看到后面的用户可以自己尝试尝试&…

Canvas笔记05:像素操作,可以对图像进行像素级别控制和处理

hello&#xff0c;我是贝格前端工场&#xff0c;最近在学习canvas&#xff0c;分享一些canvas的一些知识点笔记&#xff0c;本期分享canvas像素操作的知识&#xff0c;欢迎老铁们一同学习&#xff0c;欢迎关注&#xff0c;如有前端项目需要协助可私聊。 一、什么是像素操作 Ca…

【算法训练营】数字盒子,重编码,成绩排序(python实现)

数字盒子 问题描述 你有一个盒子&#xff0c;你可以往里面放数&#xff0c;也可以从里面取出数。 初始时&#xff0c;盒子是空的&#xff0c;你会依次做 Q 个操作&#xff0c;操作分为两类&#xff1a; 插入操作&#xff1a;询问盒子中是否存在数 x&#xff0c;如果不存在则把数…

传输频宽是啥?对网速影响有多大?

频宽&#xff0c;即WIFI频道宽度&#xff0c;又称为WIFI信道宽度&#xff0c;是WiFi Channel width的缩写。从科学的定义来说&#xff0c;Wi-Fi频道宽度&#xff0c;是指Wi-Fi无线信号在频谱上所占用的带宽大小。它决定了Wi-Fi网络的数据传输速率和稳定性&#xff0c;一般有20M…

Gas Hero Coupon NFT 概览与数据分析

作者&#xff1a;stellafootprint.network 编译&#xff1a;mingfootprint.network 数据源&#xff1a;Gas Hero Coupon NFT Collection Dashboard Gas Hero “盖世英雄” 是一个交互式的 Web3 策略游戏&#xff0c;强调社交互动&#xff0c;并与 FSL 生态系统集成&#xff0…

【java苍穹外卖项目实战一】苍穹外卖项目介绍

文章目录 1、项目介绍1、项目概述2、 产品原型3、技术选型 1、项目介绍 在开发苍穹外卖这个项目之前&#xff0c;我们需要全方位的来介绍一下当前我们学习的这个项目。接下来&#xff0c;我们将从项目简介、产品原型、技术选型三个方面来介绍苍穹外卖这个项目。 1、项目概述 …