C++ 函数详解
C++函数完全兼容C函数定义的风格,但是也做出了一些改进。一方面,C++函数允许使用缺省参数值和占位参数;另一方面,C++提供了重要的函数重载机制;另外,为了解决C中臭名昭著的宏缺陷问题,C++提供了内联函数的机制(C99之后也支持这个特性)。从C++11开始,函数定义支持后置返回类型。
一、缺省参数值
C++中可以在函数声明时为参数提供一个默认值,当函数调用时如果不提供实参,就使用这个默认值:
int func(int x = 0);
int main(int argc, char *argv[])
{
printf("func() = %d\n", func());
return 0;
}
int func(int x)
{
return x;
}
注意:默认值只在函数声明时有效,虽然函数定义中也可以写默认值,但是会被函数声明中的默认值覆盖。
函数默认参数的规则是:
- 参数的默认值需要从右向左提供
- 函数调用时如果使用了默认值,那么后续的参数都必须使用默认值
二、函数的占位参数
在C++中可以为函数提供占位参数。占位参数只有参数类型声明,但是没有参数名;这样,在函数的实现中是无法使用这个参数的:
int func(int)
{
return 0;
}
int main(int argc, char *argv[])
{
func(1); // 虽然参数无意义,但是还是要提供来通过编译。
return 0;
}
占位参数存在的意义是:兼容C语言中可能出现的不规范的写法,也可以配合默认值来一起使用。
三、函数重载
如果同一个作用域内的几个函数名称相同,但是参数列表不同,那它们就是重载函数:
void print(const char *cp);
void print(const int *begin, const int *end);
void print(const int ia[], size_t size);
函数重载有以下几个注意点:
(3.1) 函数重载必须至少要满足的条件
- 参数个数不同
- 参数类型不同
- 参数顺序不同
注意:返回值类型不同不能作为重载的条件。
(3.2) const与重载
如果函数以值传递参数,那么const
不能构成重载:
void func(int arg);
void func(const int arg); // 错误:对func的重定义
但是,如果以指针或者引用来传递参数,那么const就可以构成重载了:
void func1(int *arg);
void func1(const int *arg); // 正确:声明了新函数
void func1(const int *const arg); // 正确:声明了新函数
void func2(int &arg);
void func2(const int &arg); // 正确:声明了新函数
另一方面,const成员函数
和非const成员函数
同样可以构成函数重载:
class object
{
public:
int func();
const int func() const; // 正确:声明了新函数
};
(3.3) 重载函数的确定
编译器会以下面的顺序确定要调用的函数:
- 精确匹配:
- 实参的类型和形参完全相同。
- 实参从数组类型或者函数类型转换为对应的指针类型。
- 向实参添加
顶层const
或者从实参中移除顶层const
- 通过
const
转换实现的类型匹配。 - 通过类型提升实现的匹配。
- 通过算数类型转换实现的匹配。
- 通过类类型转换实现的匹配。
(3.4) 重载与函数指针
由于编译器需要根据重载规则去挑选与函数指针参数列表一致的函数,并且要严格地匹配函数类型与函数指针的类型,因此无法直接通过函数名得到重载函数的入口地址:
void func(int);
void func(double);
int main(int argc, char *argv[])
{
void * v = func // 错误:无法通过函数名得到函数的地址。
void(*pFunc1)(int) = func; // 正确:获得了void func(int)的入口地址。
void(*pFunc2)(double) = func; // 正确:获得了void func(double)的入口地址。
return 0;
}
(3.5) 让编译器以C语言方式编译函数
为了兼容旧有的C语言代码库,必须以C语言的编译规则来编译函数,因此需要使用如下的方式:
- 以
__cplusplus
宏来检查是否使用了C++ - 以
extern "C"
来让编译器以C语言方式编译函数
#ifdef __cplusplus
extern "C" {
#endif
void func(); // 这个函数将以C语言的方式编译。
#ifdef __cplusplus
}
#endif
四、内联函数
C++中推荐以内联函数来代替宏代码片段。C++同样以直接替换代码块的方式来处理内联函数,同时没有著名的宏缺陷问题。可以使用inline
关键字来请求编译器将函数以内联函数的方式处理 (编译器可能忽略这个请求):
inline void func() {...}
注意:
内联函数声明时
inline
必须和函数定义结合在一起,否则编译器会忽略该请求。现代的编译器会进行优化,一些函数即使没有
inline
,也可能会被内联编译。一些编译器提供了一些扩展语法,可以对函数进行强制的内联操作:
- VC:
__forceinline
- g++:
__attribute__((always_inline))
- VC:
内联函数存在一些限制:
- 不能存在循环语句
- 不能存在过多的条件判断语句
- 函数体不能太长
- 不能取函数的地址
- 内敛函数声明必须在调用这个函数之前
五、后置返回类型
C++11提供了函数的后置返回类型,用于兼容自动类型推断:
auto func() -> void {} // 等价于 void func() {}
C++ 函数详解的更多相关文章
- malloc 与 free函数详解<转载>
malloc和free函数详解 本文介绍malloc和free函数的内容. 在C中,对内存的管理是相当重要.下面开始介绍这两个函数: 一.malloc()和free()的基本概念以及基本用法: 1 ...
- NSSearchPathForDirectoriesInDomains函数详解
NSSearchPathForDirectoriesInDomains函数详解 #import "NSString+FilePath.h" @implementation ...
- JavaScript正则表达式详解(二)JavaScript中正则表达式函数详解
二.JavaScript中正则表达式函数详解(exec, test, match, replace, search, split) 1.使用正则表达式的方法去匹配查找字符串 1.1. exec方法详解 ...
- Linux C popen()函数详解
表头文件 #include<stdio.h> 定义函数 FILE * popen( const char * command,const char * type); 函数说明 popen( ...
- kzalloc 函数详解(转载)
用kzalloc申请内存的时候, 效果等同于先是用 kmalloc() 申请空间 , 然后用 memset() 来初始化 ,所有申请的元素都被初始化为 0. view plain /** * kzal ...
- Netsuite Formula > Oracle函数列表速查(PL/SQL单行函数和组函数详解).txt
PL/SQL单行函数和组函数详解 函数是一种有零个或多个参数并且有一个返回值的程序.在SQL中Oracle内建了一系列函数,这些函数都可被称为SQL或PL/SQL语句,函数主要分为两大类: 单行函数 ...
- jQuery.attr() 函数详解
一,jQuery.attr() 函数详解: http://www.365mini.com/page/jquery-attr.htm 二,jQuery函数attr()和prop()的区别: http: ...
- memset函数详解
语言中memset函数详解(2011-11-16 21:11:02)转载▼标签: 杂谈 分类: 工具相关 功 能: 将s所指向的某一块内存中的每个字节的内容全部设置为ch指定的ASCII值, 块的大 ...
- CreateFile函数详解
CreateFile函数详解 CreateFile The CreateFile function creates or opens the following objects and returns ...
- MYSQL常用内置函数详解说明
函数中可以将字段名当作变量来用,变量的值就是该列对应的所有值:在整理98在线字典数据时(http://zidian.98zw.com/),有这要一个需求,想从多音字duoyinzi字段值提取第一个拼音 ...
随机推荐
- Lambda函数到底是个什么
1 什么是Lambda函数 lambda函数是指简单的代码片段,通常认为是不值得命名的函数,它不能重复使用,能方便程序员使用,增强代码可读性,降低代码出错概率. [ 捕获列表 ] (参数) -> ...
- 笔记软件->"Typora"
笔记软件->"Typora" 1 下载地址 www.typora.io 2 傻瓜安装后添加自定义样式 由于用户目录不同Users为当前系统用户名字 打开C:\Users\Ad ...
- 洛谷P1638 逛画展 题解 尺取法/双指针/队列
题目链接:https://www.luogu.com.cn/problem/P1638 题目大意: 给你一个长度为 \(n (\le 10^6)\) 的数组,数组中每个元素的范围在 \(1\) 至 \ ...
- spring boot学习笔记(2)
Spring boot集成mybatis的三种方式 一.XML文件 在pom文件里面引入mybatis和数据库的依赖 在application.properties中加入数据源配置 其他和ssm配置完 ...
- DevOps is Hard、DevSecOps is Even Harder . --- Enterprise Holdings
Enterprise Holdings. 的IT团队超过2000人,在2018年的演讲中介绍了Enterprise Holdings的DevOps是如何转型的.我们通过打造一个不只包涵了pipelin ...
- Java Linked集合的简单介绍和常用方法的使用
LinkedList的简单介绍 java.util.LinkedList 集合数据存储的结构是链表结构.LinkedList是一个双向链表在实际开发中,对一个集合元素的添加和删除,经常涉及到首尾操作, ...
- Java Collection集合概述及其常用方法
Collection集合概述 Java数组的长度是固定的,为了使程序能够方便地存储和操作数目不固定的一组数据,JDK类库提供了Java集合 与数组不同的是,集合中不能存放基本类型数据,而只能存放对象的 ...
- 怎样使用七牛云CDN加速并绑定阿里云域名
昨天晚上在某个群里看到群友问,七牛云能不能绑定自己的域名作为静态资源文件的前缀,忽然想起来我已经有快两年时间没有登录过我的七牛云账号了,不禁老脸一红,这是有多久没有自己前后端都弄了,幸好还没有老年痴呆 ...
- 【Linux】---Linux系统下各种常用命令总结
在Linux系统下,“万物皆文件”,之所以强调在强调这个概念,是因为很多人已经习惯了win系统下找找点点得那种方式和思维,因此总是会觉得linux系统下很多指令既复杂又难记.其实都是一样得东西,只是w ...
- 两个大数相乘 - 高精度FFT
HDU 1402 A * B Problem Plus Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (J ...