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字段值提取第一个拼音 ...
随机推荐
- beta 1/2 阶段中间产物提交入口
此作业要求参见:https://edu.cnblogs.com/campus/nenu/2019fall/homework/9918 git地址:https://e.coding.net/Eustia ...
- 【题解】PKUWC2018简要题解
[题解]PKUWC2018简要题解 Minimax 定义结点x的权值为: 1.若x没有子结点,那么它的权值会在输入里给出,保证这类点中每个结点的权值互不相同. 2.若x有子结点,那么它的权值有p的概率 ...
- java poi ppt 接口的基本操作
依赖 在 pom.xml中增加以下依赖 <dependency> <groupId>org.apache.poi</groupId> <artifactId& ...
- java中高级并发SPI机制
Java SPI 实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制. 适用于:调用者根据实际使用需要,启用.扩展.或者替换框架的实现策略. 要使用Java SPI,需要遵循如下约定 ...
- (二)Angular+spring-security-cas前后端分离(基于ticket代码实现
一.前端实现 1.1.路由守卫(用于拦截路由认证) import { Injectable, Inject } from "@angular/core"; import { Can ...
- Http GetPost网络请求
using Newtonsoft.Json; using System; using System.Collections.Generic; using System.IO; using System ...
- 【Java基础总结】网络编程
网络编程 InetAddress tcp udp
- 【C&数据结构】---关于链表结构的前序插入和后序插入
刷LeetCode题目,需要用到链表的知识,忽然发现自己对于链表的插入已经忘得差不多了,以前总觉得理解了记住了,但是发现真的好记性不如烂笔头,每一次得学习没有总结输出,基本等于没有学习.连复盘得机会都 ...
- numpy基本知识
"""np.arrayobject 数组或嵌套的数列dtype 数组元素的数据类型,可选copy 对象是否需要复制,可选order 创建数组的样式,C为行方向,F为列方向 ...
- limit_choices_to
# 多对多 老师 teachers = models.ManyToManyField(verbose_name='任课老师', to='UserInfo', related_name="ab ...