C++ 函数的介绍

目录
  • 一、基础
  • 二、参数
  • 三、返回类型
  • 四、函数重载与解析
  • 五、内联函数
    • 1.constexpr函数
  • 六、函数指针
  • 七、思考
    • 1、我们常常会见到如下代码,是由什么作用?
    • 2、可以用别名定义一个函数类型吗?

一、基础

函数:封装了一段代码,可以在一次执行过程中被反复调用,包含函数头和函数体;

函数头:

  • 函数名称(标识符),用于后续的调用;
  • 形式参数,代表函数的输入参数;
  • 返回类型,函数执行完成后返回结果的类型;

函数体:一个语句块(block),包含具体的计算逻辑;

函数的声明与定义:

  • 函数声明只包含函数头,不包含函数体,通常在头文件中;
  • 函数声明可以出现多次,定义通常出现一次(也有例外);

函数调用:

  • 需要提供函数名与实际参数;
  • 实际参数拷贝初始化给形式参数;
  • 返回值会拷贝给函数的调用者;
  • 栈帧结构(可自行了解下);

二、参数

对于非模板函数来说,其每个形参都有确定的类型,但形参可以没有名称;

实参到形参的拷贝顺序是不确定的;

函数的形参的传递一般分为:传值、传址、传引用;

变长参数的定义:

1、使用initializer_list传递

#include <initializer_list>
void fun(std::initializer_list<int> a){}

int main
{
 fun({1, 2, 3, 4})
}

注意:该方法只能传递类型相同的变长参数;

  • 可变长度模板参数
  • 使用省略号表示形式参数(一般不使用)

函数的缺省实参注意点:

1、如果某个形参具有缺省参数,那么它右侧的形参都必须具有缺省实参;

void fun(int x=1, int y=2){} // 这里y必须给定缺省值

2、具有缺省实参的函数调用时,传入实参按照从左到右的顺序进行匹配;

3、在一个翻译单元中,每个形参的缺省实参只能定义一次;

4、缺省实参为对象时,实参的缺省值会随对象值的变化而变化;

main函数的版本:

  • 无形参版本(一般使用)
  • 带形参版本
int main(int argc, char *argv[]) {}

argc是非负数,表述传入参数个数,argv是一个指针指向传输参数的数组头。

三、返回类型

返回类型的几种书写方式:

经典方法:位于函数头的前部,也是最常规的写法;

C++11引入的方式:位于函数头的尾部;

auto fun(int x) -> int
{
 return x*2;
}

C++14引入的方式:返回类型的自动推导;

auto fun(int a)
{
 return a;  // 会根据return语句进行推导
}

四、函数重载与解析

函数重载:使用相同的函数名定义多个函数,每个函数具有不同的参数列表;

注意:不能基于不同的返回类型进行重载;

名称查找:

  • 分为限定查找和非限定查找:有无限定在某个作用域中;
  • 非限定查找会进行域的逐级查找——名称隐藏;
  • 查找通常只会在已声明的名称集合中进行;

重载解析:在名称查找的基础上进一步选择合适的调用函数;

  • 过滤不能被调用的版本:参数个数不对、无法将实参转为形参、实参不满足形参的限制条件;

五、内联函数

定义:将比较简单的函数逻辑展开到调用函数的部分,避免栈帧销毁,提升性能;

关键字:inline,如果一个函数在多个翻译单元展开,加入这个关键字可以避免重复定义;

1.constexpr函数

定义:之前有介绍常量表达式时用到了该关键字,现在对于函数也可以用该关键字;

作用:使得函数在编译器被执行,当然在有变量情况下也可在运行期执行;

constexpr int fun(int x){
    // int y; std::cin >> y;  会报错,该语句需要用户传入参数,只能在运行期执行
 return x * 2;
}

int main
{
 constexpr int x = fun(2);  // 编译器会翻译成 move eax 4, 去掉constexpr也可以
 return x;
}

注意:constexpr函数中的语句必须是可以在编译器执行的语句;

拓展:关键字consteval(C++20引入),函数只能在编译器执行;

六、函数指针

作用:可以用于高阶函数中,将函数指针作为参数;

代码案例:

int add(x) { return x + 1};
using T = int(int);
int fun(K* F, int x)
{
 int tmp = (*F)(x);
 return tmp * 2;
}

int main
{
 std::cout << fun(&add, 50) << std::endl;
}

说明:这就是用函数指针定义的一个高阶函数,在之后的很多高阶函数、泛型算法中也是这样的用法;

注意:当函数对象进行赋值或者返回值时,返回的是一个函数指针类型的对象;

七、思考

1、我们常常会见到如下代码,是由什么作用?

extern "C"
int fun(int x, int y)
{
    return x + y;
}

C语言对于函数是不能重载的,当用C调用C++程序时,往往找不到C++编译后的函数名,可通过如上代码定义一个函数为C类型函数;

2、可以用别名定义一个函数类型吗?

using X = int[3];
X a;    // 这是定义了一个数组,同int a[3]

using X = int(int);
X fun;    // 这是定义了一个int返回类型的函数

函数也是有类型的,可以用别名定义,并且函数类型不包含形参名称,并且只能声明,不能定义;

总结:

本篇主要介绍了函数的基础概念以及一些特殊的函数方法和类型。重点需要注意的就是函数重载以及函数指针,这个在后续的模板以及泛型编程都会用到;下一篇将讨论C++中的内存,也是最重要的一个部分。

到此这篇关于C++ 函数的介绍的文章就介绍到这了,更多相关C++函数内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++中atof 函数的介绍

    目录 一.atof 函数 二.atof 函数函数实战 一.atof 函数 在 stdlib.h 中 atof 函数,可用于将 char 字符串转为 float / double 浮点数类型, 语法如下: /* *描述:将一个char类型转为浮点数double * *参数: * [in] str:字符串类型 * *返回值:返回char类型对应的浮点数double */ double atof ( const char * str ); 二.atof 函数函数实战 #include "stdafx.

  • C++ 函数指针详细总结

    目录 1.函数指针 1.1 获取函数地址 1.2 声明函数指针 1.3 函数指针传参 1.4 调用函数 2.函数指针进阶 1.函数指针 函数指针顾名思义,就是指向函数的指针. 和数据类似,C++当中函数也有地址,函数的地址是存储函数机器语言代码的内存地址.我们可以将另外一个函数的地址作为参数传入函数,从而实现函数的灵活调用. 1.1 获取函数地址 获取函数地址的方法非常简单,只要使用函数名(后面不跟参数和括号)即可.比如我们有一个函数叫做think,那么think()是调用函数拿到结果,而thi

  • C++ 内联函数详解

    目录 一.C++ 内联函数 1.内联 2.语法 3.评价 总结 一.C++ 内联函数 1.内联 C 语言中有宏函数的概念.宏函数的特点是内嵌到调用代码中去,避免了函数调用的开销.但是由于宏函数的处理发生在预处理阶段,缺失了语法检测和有可能带来的语意差错. 2.语法 C++提供了 inline 关键字,实现了真正的内嵌. 宏函数 VS inline函数 #include <iostream> #include <string.h> using namespace std; //优点:

  • C++内联函数详情

    内联函数是C++当中为了提高程序运行效率的设计,老实讲我没有在其他语言当中看到类似的设计.它和常规函数之间的主要区别不在于编写的方式,而是在于C++编译器会将内联函数组合到程序当中执行. 要解释这个过程会稍稍有些复杂,我们需要从编译的过程说起.对于编译型语言而言,编译器做的事情是把人类写出来人能读懂的代码翻译成机器能够识别.执行的机器语言,一般是一串十六进制的指令.随后计算机逐步执行这些指令,完成我们想要的功能. 当我们调用函数时,其实本质上是指令跳转,先记录下当前运行的指令位置,跳转到函数所在

  • 详解C++函数类型与重载函数

    目录 1. 2. 问题: 总结 1. 首先对重载函数,明确函数的返回类型不能决定重载函数的类别,即 int F(int ,int) ://一个返回 int 类型的函数 void F(int ,int)://一个无返回值的函数 //两者形参列表相同,返回值类型不同,但两者不构成重载函数 2. 注意形参列表中的默认值,使用含默认参数的重载函数时可能会产生二义性.例: int a = 0; int Max(int,int); int Max(int,int,int = 0); //则对Max(3,5)

  • C++中replace() 函数的基本用法

    目录 replace算法: 用法一:用str替换指定字符串从起始位置pos开始长度为len的字符 用法二: 用str替换 迭代器起始位置 和 结束位置 的字符 用法三: 用substr的指定子串(给定起始位置和长度)替换从指定位置上的字符串  用法四:string转char*时编译器可能会报出警告,不建议这样做 用法五:string转char*时编译器可能会报出警告,不建议这样做 用法六:string转char*时编译器可能会报出警告,不建议这样做 用法七:string转char*时编译器可能会

  • oracle中length、lengthb、substr、substrb函数用法介绍

    我记得我曾经在开发form的时候犯过这样一个错误,对于form中的某个字段,对应于数据库中某张表的字段,假设在数据库中这个字段一般也就用到20个汉字的长度,后来我在开发form的时候,设置item类型长度的时候,我惯性的设置成了50byte,想着就算是20个汉字,最多也就占40个byte长度嘛.可是,就因为这一个想当然,结果出现错误了,后来发现数据库字符集编码是utf8,那么应该设置为60.从那以后,每次涉及到给字段设置长度的时候,我都会特别注意下,到底是啥编码. 在oracle中,比较常见的可

  • Numpy中stack(),hstack(),vstack()函数用法介绍及实例

    1.stack()函数 函数原型为:stack(arrays,axis=0),arrays可以传数组和列表.axis的含义我下面会讲解,我们先来看个例子,然后我会分析输出结果. import numpy as np a=[[1,2,3], [4,5,6]] print("列表a如下:") print(a) print("增加一维,新维度的下标为0") c=np.stack(a,axis=0) print(c) print("增加一维,新维度的下标为1&qu

  • python 的numpy库中的mean()函数用法介绍

    1. mean() 函数定义: numpy.mean(a, axis=None, dtype=None, out=None, keepdims=<class numpy._globals._NoValue at 0x40b6a26c>)[source] Compute the arithmetic mean along the specified axis. Returns the average of the array elements. The average is taken over

  • Python3-异步进程回调函数(callback())介绍

    废话不多说,大家之家看代码吧! #异步 ''' 举例: 你喊你朋友吃饭,你朋友正忙, 如果你一直在那等他,等你朋友忙完了,你们一块去.--同步调用 你喊你朋友吃饭,你朋友正忙, 如果你自己做你自己的事,你朋友忙完,找到你,一块去吃饭.--异步调用 ''' # from bs4 import BeautifulSoup from multiprocessing import Process,Pool import os import time #子进程任务 def download(): prin

  • tensorflow常用函数API介绍

    摘要:本文介绍了tensorflow的常用函数. 1.tensorflow常用函数 TensorFlow 将图形定义转换成分布式执行的操作, 以充分利用可用的计算资源(如 CPU 或 GPU.一般你不需要显式指定使用 CPU 还是 GPU, TensorFlow 能自动检测.如果检测到 GPU, TensorFlow 会尽可能地利用找到的第一个 GPU 来执行操作. 并行计算能让代价大的算法计算加速执行,TensorFlow也在实现上对复杂操作进行了有效的改进.大部分核相关的操作都是设 备相关的

  • python 高阶函数简单介绍

    把函数作为参数传入,这样的函数称为高阶函数,高阶函数是函数式编程的体现.函数式编程就是指这种高度抽象的编程范式. 1.体验高阶函数 在Python中,abs()函数可以完成对数字求绝对值计算. abs(-10) # 10 round()函数可以完成对数字的四舍五入计算. round(1.2) # 1 round(1.9) # 2 需求:任意两个数字,按照指定要求整理数字后再进行求和计算. 方法1 def add_num(a, b): return abs(a) + abs(b) result =

  • C语言中qsort函数的介绍与用法实例

    目录 一.qsort函数是什么 二.使用qsort排序-以升序为例 1.整形数组排序 2.字符数组排序 3.字符指针数组排序 4.结构体数组排序 5.浮点型数组排序 三.使用冒泡排序思想模拟实现qsort函数 1.什么是冒泡排序: 2.冒泡排序代码 3. 使用冒泡排序思想模拟实现qsort函数 总结 一.qsort函数是什么 我们可以使用  搜索库函数网址或者MSDN软件进行查找. qsort()函数:快速排序的函数  -引用stdlib.h头文件 参数说明: void qsort ( void

  • Python 循环函数详细介绍

    目录 一.循环函数 1.for循环 2.while循环 3.中断循环 二.循环设计 1.range() 2.enumerate() 3.zip() 三.循环对象 1.什么是循环对象 2.迭代器 3.生成器 4.表推导 一.循环函数 1.for循环 for循环需要预先设定好循环的次数(n),然后执行隶属于for的语句n次. 基本构造是 for 元素 in 序列: statement 举例来说,我们编辑一个叫forDemo.py的文件 for a in [3,4.4,'life']: print a

  • JavaScript的function函数详细介绍

    通过函数来封装任意多条语句,而且可以在任何地方.任何时间调用执行. 而我们的JavaScript脚本语言比较特殊,相对于C语言,它的参数是不需要数据类型加持的.返回值return,我就不过多描述,他是和 C语言通的,如果没写他就会自动返回undefined function fun(x,y){ } //写成这样就可以声明一个函数 以我的理解他就是以对象的形式来传入参数,通过对象的各项属性值(引用类型的值),来作为我的实际参数, 例如我有以下做法: function fun(x, y) { //

随机推荐