//函数扩展--内联函数 inline
#include<iostream>
using namespace std; /*
c++中const常量可以替代宏常数定义 如:
const int A = 3; 近似于 #define A 3
但是 const无法替代宏代码片段
c++中推荐使用内联函数替代宏代码片段
c++中使用inline关键字声明内联函数
内联函数声明时inline关键字必须和函数定义结合在一起,否则c++编译器会忽略内联请求 c++编译器可以将一个函数进行内联编译
被c++编译器内联编译的函数叫做内联函数
内联函数在最终生成的代码中是没有定义的
c++编译器会直接将函数体插入到函数调用的地方(跟宏替换有点相似)
内联函数没有普通函数调用时的额外开销(压栈,跳转,返回) 内联函数是一种特殊的函数,具有普通函数的特征(参数检查,返回类型等)
内联函数是对c++编译器的一种请求,因此c++编译器可能拒绝这种请求
内联函数由c++编译器处理,直接将编译后的函数体插入调用的地方
宏代码片段由预处理器处理,进行简单的文本替换,没有编译过程 现在c++编译能够进行编译优化,因此一些函数计时没有inline声明,也可能被编译器内联编译 c++中的内联编译限制(c++编译器有可能拒绝这种内联请求)
①不能存在任何形式的循环语句
②不能存在过多的条件判断语句
③函数体不能过于庞大
④不能对函数进行取地址操作(内联函数在最终生成的代码中是没有定义的,c++编译器会直接将函数体插入到函数调用的地方)
⑤函数内联声明必须在调用语句之前 c++编译器对于内联函数的限制并不是绝对的,内联函数相对于普通函数的优势只是省去了函数调用时的压栈,跳转和返回的开销
因此,当函数的执行开销远大于压栈,跳转,返回所用的开销是,那么内联函数将毫无意义,所以内联函数适用于简单的函数 结论:
内联函数在编译时直接将函数体插入函数调用的地方
inline只是一种请求,c++编译器不一定允许这种请求
内联函数省去了普通函数调用时的压栈,跳转和返回的开销
*/ inline int PrintA(int a, int b); inline int PrintA(int a, int b){
return a < b ? a : b;
} #define Func(a,b) (a)<(b)?(b):(a) void main(){
int a1 = , b1 = , c1 = ;
//内联函数
c1 = PrintA(++a1, b1);
//相当于
/*{
return a1 < b1 ? a1 : b1;
}*/
printf("a1=%d\n", a1);//打印2
//宏定义的副作用
int a2 = , b2 = , c2 = ;
c2 = Func(++a2, b2);//相当于 (++a2)>(b2)?(++a2):(b2)
printf("a2=%d\n", a2);//打印3 system("pause");
}
//函数扩展--默认参数
#include<iostream>
using namespace std; //默认参数的规则:
//只有参数列表后面部分的参数才可以提供默认参数
//一旦一个函数中使用了默认参数,那么这个参数后面的所有参数都必须使用默认参数值
//例如:
/*
void PrintT(int a=3, int b){
printf("a=%d;b=%d\n", a, b);
}
*/
//报错 error C2548: “PrintT”: 缺少参数 2 的默认参数 //函数的默认参数
void PrintA(int a, int b=){
printf("a=%d;b=%d\n",a,b);
} void main(){
PrintA(, );//打印 a=1;b=2
PrintA();//打印 a=1;b=3
system("pause");
}
//函数扩展--函数占位符 (了解)
#include<iostream>
using namespace std; /*
函数占位符
占位参数只有参数类型声明,而没有参数名声明
一般情况下,在函数体内部无法使用占位参数
所以根本没有实际用处
一般的应用场景就是操作符重载里表示后置操作符
*/
int Fuc(int a, int b, int){
return a + b;
} void main(){
//Fuc(1, 2);
//报错 error C2660: “Fuc”: 函数不接受 2 个参数
//说明:函数占位符虽然没有实际作用,但是还必须传一个参数
int c = ;
c=Fuc(, ,);
printf("c=%d\n", c);
system("pause");
}

C++ 函数的扩展①的更多相关文章

  1. ES6 - Note3:数组、对象与函数的扩展

    一.数组的扩展,ES6在数组扩展了一些API,以实现更多的功能 1.Array.from:可以将类数组和可遍历的数据结构转换成真正的数组,如下所示 var a = { '0':1,'1':1,leng ...

  2. ES6入门之函数的扩展

    函数js原有的: 属性:arguments[].caller(调用该函数的引用,注意与callee分别开,callee指的是调用函数本身经常在递归中出现).length(形参个数).prototype ...

  3. 编译器对C++ 11变参模板(Variadic Template)的函数包扩展实现的差异

    编译器对C++ 11变参模板(Variadic Template)的函数包扩展实现的差异 题目挺绕口的.C++ 11的好东西不算太多,但变参模板(Variadic Template)肯定是其中耀眼的一 ...

  4. sqlserver 只有函数和扩展存储过程才能从函数内部执行

    一个SQLServer的自定义函数中调用一个自定义的存储过程,执行此函数后发出如下提示:“只有函数和扩展存储过程才能从函数内部执行". 原因:函数只能使用简单的sql语句,逻辑控制语句,复杂 ...

  5. ES6_入门(6)_函数的扩展

    // 2017/7/22 /*ES6函数的扩展*/ //ES6 之前,不能直接为函数的参数指定默认值,只能采用变通的方法. function log(x, y) { y = y || 'World'; ...

  6. ES6的新特性(7)——函数的扩展

    函数的扩展 函数参数的默认值 基本用法 ES6 之前,不能直接为函数的参数指定默认值,只能采用变通的方法. function log(x, y) { y = y || 'World'; console ...

  7. php 函数追踪扩展 phptrace

    php 函数追踪扩展 phptrace 介绍 phptrace 是一个低开销的用于跟踪.分析 php 运行情况的工具. 它可以跟踪 php 在运行时的函数调用.请求信息.执行流程.并且提供有过滤器.统 ...

  8. ES6学习(二):函数的扩展

    chapter07 函数的扩展 7.1 函数默认值 7.1.1 参数默认值简介 传统做法的弊端(||):如果传入的参数相等于(==)false的话,仍会被设为默认值,需要多加入一个if判断,比较麻烦. ...

  9. ES6学习笔记(6)----函数的扩展

    参考书<ECMAScript 6入门>http://es6.ruanyifeng.com/ 函数的扩展 函数的默认值 : ES6可以为函数指定默认值 (1)指定默认值的两种方式 a.函数参 ...

  10. ES6__函数的扩展

    /** * 函数的扩展 * 1 为函数参数指定默认值 * 2 函数的 rest 参数 * 3 箭头函数 */ // ------------------------------------------ ...

随机推荐

  1. SQL Server 之 与 OVER() 函数

    在SQL SERVER 2005/2008支持两种排名开窗函数和聚集开窗函数. 一. OVER() 函数 语法结构:OVER( [ PARTITION BY ... ] [ ORDER BY ... ...

  2. Generator [ˈdʒenəreɪtə(r)] 函数结构

    Generator函数是ES6新增的一种异步编程方案. 说明:Generator函数指的是一种新的语法结构,是一个遍历器对象生成器,它内部可以封装多个状态,非常适合用于异步操作. Generator函 ...

  3. Java之开发工具(1) - Eclipse 如何设置注释的模板

    最常用的注释就是对类的说明和方法的说明,关于这类代码的注释方式,在Eclipse中可以这样进行设置: windows---preferences...---java--code style--code ...

  4. JavaWeb 获取请求网络协议、IP、端口号、项目根路径

      JavaWeb 获取请求网络协议.IP.端口号.项目根路径 CreateTime--2018年6月1日16点32分 Author:Marydon 1.需求 在项目中,需要使用Java向本程序发送r ...

  5. 模式的秘密-代理模式(2)-JDK动态代理

    代理模式-动态代理 (1) (2) 代码实践动态代理: 第一步:被代理类的接口: package com.JdkProxy; public interface Moveable { void move ...

  6. 18-spring学习-利用Annotation配置AOP

    之前是通过配置完成aop操作,如果自己写的话,太麻烦了,可以使用基于annotation的配置完成. 第一步:打开AOP的annotation支持 加上一句话: <context:annotat ...

  7. C#:数据库操作(待补充)

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.D ...

  8. 分别用C和C++来实现一个链栈

    下面通过分别用C和C++来实现一个链栈(链表实现),从中体会数据封装抽象的思想: C语言实现:  C++ Code  1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 ...

  9. redis.windows.conf各项配置参数介绍 (九)

    # 默认情况下,redis不是在后台模式运行的,如果需要在后台进程运行,把该项的值更改为yes,默认为no daemonize:是否以后台daemon方式运行 # 如redis服务以后台进程运行的时候 ...

  10. unity, instantiate一个实例后,先指定parent,再指定position

    instantiate一个实例后,先指定parent,再指定position,才能保证position正确,如果先指定position再指定parent,则position会错误.