C++自定义函数详解

个人主页: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.rhkb.cn/news/255275.html

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

相关文章

Acwing---837. 连通块中点的数量

连通块中点的数量 1.题目2.基本思想3.代码实现 1.题目 给定一个包含 n n n个点&#xff08;编号为 1 ∼ n 1∼n 1∼n&#xff09;的无向图&#xff0c;初始时图中没有边。 现在要进行 m m m 个操作&#xff0c;操作共有三种&#xff1a; C a b&#xff0c;在点 a 和点 b …

计算机网络基本知识(二)

文章目录 概要分层为什么分层怎么分层&#xff1f;1.实体2.协议3.服务 分层基本原则正式认识分层详细例子解释 总结 概要 分层知识&#xff1a;概念理解 分层 为什么分层 大致以上五点 为了解决上面的问题&#xff08;复杂&#xff09; 大问题划分为小问题 怎么分层&#…

【后端高频面试题--Mybatis篇】

&#x1f680; 作者 &#xff1a;“码上有前” &#x1f680; 文章简介 &#xff1a;后端高频面试题 &#x1f680; 欢迎小伙伴们 点赞&#x1f44d;、收藏⭐、留言&#x1f4ac; 后端高频面试题--Mybatis篇 什么是Mybatis&#xff1f;Mybatis的优缺点&#xff1f;Mybatis的特点…

提升MySQL访问性能

1. 读写分离 设置多个从数据库&#xff0c;从数据库可能在多个机器中。写操作在主数据库进行主数据库提供数据的主要依据 缓解了MySQL的读压力。 主从复制原理图如下 如果对于读操作有一致性要求&#xff0c;那么读操作去主数据库即可。 2. 连接池 因为一个请求必须要…

C语言-----自定义类型-----结构体枚举联合

结构体和数组一样&#xff0c;都是一群数据的集合&#xff0c;不同的是数组当中的数据是相同的类型&#xff0c;但是结构体中的数据类型可以不相同&#xff0c;结构体里的成员叫做成员变量 结构体类型是C语言里面的一种自定义类型&#xff0c;我们前面已经了解到过int,char,fl…

SpringSecurity(19)——OAuth2客户端信息存储

ClientDetailsService public interface ClientDetails extends Serializable {String getClientId();//客户端idSet<String> getResourceIds();//此客户端可以访问的资源。如果为空&#xff0c;则调用者可以忽略boolean isSecretRequired();//验证此客户端是否需要secr…

LeetCode---383周赛

题目列表 3028. 边界上的蚂蚁 3029. 将单词恢复初始状态所需的最短时间 I 3030. 找出网格的区域平均强度 3031. 将单词恢复初始状态所需的最短时间 II 一、边界上的蚂蚁 这题没什么好说的&#xff0c;模拟就行&#xff0c;本质就是看前缀和有几个为0。 代码如下 class S…

IDEA创建SpringBoot+Mybatis-Plus项目

IDEA创建SpringBootMybatis-Plus项目 一、配置Maven apache-maven-3.6.3的下载与安装&#xff08;详细教程&#xff09; 二、创建SpringBoot项目 在菜单栏选择File->new->project->Spring Initializr&#xff0c;然后修改Server URL为start.aliyun.com&#xff0c…

javaEE - 23( 21000 字 Servlet 入门 -1 )

一&#xff1a;Servlet 1.1 Servlet 是什么 Servlet 是一种实现动态页面的技术. 是一组 Tomcat 提供给程序猿的 API, 帮助程序猿简单高效的开发一个 web app. 构建动态页面的技术有很多, 每种语言都有一些相关的库/框架来做这件事&#xff0c;Servlet 就是 Tomcat 这个 HTTP…

JAVA设计模式之建造者模式详解

建造者模式 1 建造者模式介绍 建造者模式 (builder pattern), 也被称为生成器模式 , 是一种创建型设计模式. 定义: 将一个复杂对象的构建与表示分离&#xff0c;使得同样的构建过程可以创建不同的表示。 **建造者模式要解决的问题 ** 建造者模式可以将部件和其组装过程分开…

前端小案例——动态导航栏文字(HTML + CSS, 附源码)

一、前言 实现功能: 这案例是一个具有动态效果的导航栏。导航栏的样式设置了一个灰色的背景&#xff0c;并使用flex布局在水平方向上平均分配了四个选项。每个选项都是一个li元素&#xff0c;包含一个文本和一个横向的下划线。 当鼠标悬停在选项上时&#xff0c;选项的文本颜色…

烟火可禁却难禁,灵境难及终将及

现实痛点 2024年1月30日&#xff0c;贵阳市发生了一件令人痛心的事&#xff0c;有人在小区内放烟花导致失火&#xff0c;一男子具备足够的消防安全知识&#xff0c;知道如何使用消防栓却因设施不合格接不上消防栓&#xff0c;接上了又没水&#xff0c;消防员来也束手无策&…

【大厂AI课学习笔记】【1.5 AI技术领域】(10)对话系统

对话系统&#xff0c;Dialogue System&#xff0c;也称为会话代理。是一种模拟人类与人交谈的计算机系统&#xff0c;旨在可以与人类形成连贯通顺的对话&#xff0c;通信方式主要有语音/文本/图片&#xff0c;当然也可以手势/触觉等其他方式 一般我们将对话系统&#xff0c;分…

Docker 容器网络:C++ 客户端 — 服务器应用程序。

一、说明 在下面的文章中&#xff0c; 将向您概述 docker 容器之间的通信。docker 通信的验证将通过运行 C 客户端-服务器应用程序和标准“ping”命令来执行。将构建并运行两个单独的 Docker 映像。 由于我会关注 docker 网络方面&#xff0c;因此不会提供 C 详细信息。…

【Fabric.js】监听画布or元素的点击、选中、移动、添加、删除销毁、变形等各事件

在fabric使用过程中&#xff0c;如果想要玩各种花样&#xff0c;那么fabric的事件监听是一定、必须、肯定要掌握&#xff01;&#xff01;&#xff01; 例子就用vue项目组件里的代码&#xff0c;fabric的使用跟vue、react、angular之类的框架都没任何关系&#xff01; 并且本de…

第三百一十八回

文章目录 1. 概念介绍2. 使用方法2.1 本地缓冲2.2 服务器缓冲3. 示例代码4. 内容总结我们在上一章回中介绍了"如何让输入键盘不遮挡屏幕"相关的内容,本章回中将介绍如何有效地缓冲网络图片.闲话休提,让我们一起Talk Flutter吧。 1. 概念介绍 我们在本章回中介绍的…

部署tomcat

1 idea打包完&#xff0c;找到target目录下的网站 我的叫做 test.webSYS 2 linux 安装 tomcat 后 修改conf下的server.xml中的Connector的port <Connector port"9001" protocol"HTTP/1.1"connectionTimeout"20000"redirectPort"8443…

1、学习 Eureka 注册中心

学习 Eureka 注册中心 一、创建 Eureka 微服务0、SpringBoot 和 SpringCloud 版本1、引入 Eureka 服务端依赖2、启动类加 EnableEurekaServer 注解3、配置 yaml 文件&#xff0c;把 Eureka 服务注册到 Eureka 注册中心4、访问 Eureka 服务端&#xff0c;查看注册中心的服务列表…

华为配置访客接入WLAN网络示例(MAC优先的Portal认证)

配置访客接入WLAN网络示例&#xff08;MAC优先的Portal认证&#xff09; 组网图形 图1 配置WLAN MAC优先的Portal认证示例组网图 业务需求组网需求数据规划配置思路配置注意事项操作步骤配置文件 业务需求 某企业为了提高WLAN网络的安全性&#xff0c;采用MAC优先的外置Portal认…

【力扣 51】N 皇后(回溯+剪枝+深度优先搜索)

按照国际象棋的规则&#xff0c;皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。 n 皇后问题 研究的是如何将 n 个皇后放置在 nn 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 给你一个整数 n &#xff0c;返回所有不同的 n 皇后问题 的解决方案。 每一种…