C++中的函数知识点大全

C++中的函数知识点大全 函数的定义不能嵌套但调用可以嵌套在函数调用时如某一默认参数要指明一个特定值则有其之前所有参数都必须赋值赋默认实参时 一旦某个形参被赋予了默认值它后面的所有形参都必须有默认值因为设置默认参数的顺序是自右向左且注意默认值不可以是局部变量函数参数的默认值可以是表达式如果在函数定义时设置了默认参数则就不能在函数声明时再次设置反之亦然函数只有一个 返回值除void类型函数函数调用可以出现在执行语句中也可以出现在表达式中甚至还可以作为一个函数的实参但不可作为函数的形参因为函数返回值存在寄存器中, 没有地址, 不能作为形参函数是一种特殊的数据类型正确当函数不是void类型且函数体内没有return语句时此时函数的返回值与返回类型相同但内容却是随机的一个值C所有的函数本质上都是外部函数可延申至其他文件中使用故extern关键字可省略如果函数的形参是指向普通变量的指针变量实参只能用指向普通变量的指针不能用指向const变量的指针反之则都可以用当函数自变量个数不确定时系统不自动检测自变量函数三种传参① 值传递会为形参重新分配内存空间 将实参的值拷贝给形参形参的改变不会影响实参的值函数被调用结束后形参被释放。② 地址的传递形参为指针变量将实参的地址传递给函数可以在函数中改变实参的值。调用时为形参指针变量分配内存结束时释放指针变量。③ 引用传递不会为形参重新分配内存空间形参只是实参的别名形参的改变只会影响实参的值函数调用结束后形参不会被释放。二函数的使用建立自定义函数调用时只需要明白函数的功能即可故提高了程序的可读性sizeof 返回的值表示的含义如下单位字节​ 数组 —— 编译时分配的数组空间大小​ 指针 —— 存储该指针所用的空间大小存储该指针的地址的长度是长整型应该为 4 ​ 类型 —— 该类型所占的空间大小​ 对象 —— 对象的实际占用空间大小​ 函数 —— 函数的返回类型所占的空间大小。函数的返回类型不能是 voidsizeof(float)是(整型)类型表达式Math.floor() 表示向下取整返回double类型​ Math.ceil() 表示向上取整返回double类型​ Math.round() 四舍五入返回int类型用户可以重载不能重定义标准库函数若如此该函数将失去原有含义但若已包含标准库头文件及相关命名空间则系统不允许用户重新定义标准库函数因为两个相同作用域内的函数 如果除了返回值类型外 的函数要素都相同 那么编译器会报重定义错误函数返回值作为右值被const修饰无效故此时const相当于没修饰如果参数类型不一致则函数调用时按形参类型隐式类型转换实参main函数默认返回一个int类型的值将一个字符串传递到函数中传递的是地址则函数形参既可以用字符数组又可以用指针变量函数的返回值可以是引用类型且函数返回引用可以作为左值函数的返回类型可以是结构体类型这时函数将返回一个结构体对象所有的函数在定义它的程序中都是可见的三预处理命令包括宏定义预处理命令行不能以分号结尾预处理命令行可以出现在程序的最后一行预处理命令行作用域是从出现位置开始到源程序文件末尾凡是以#号开头的行不一定都为编译预处理命令行在源文件的一行上不可以有多条预处理命令预处理不做语法检查C在编译前由预处理器对预处理命令进行处理故在编译前被执行编译时进行语法分析宏替换不占用程序的运行时间只占编译时间四内联函数与宏的区别内联函数在运行时可调试而宏定义不可以;编译器会对内联函数的参数类型做安全检查或自动类型转换同普通函数而宏定义则不会内联函数可以访问类的成员变量宏定义则不能在类中声明同时定义的成员函数自动转化为内联函数。五函数与宏的区别宏做的是简单的字符串替换(注意是字符串的替换,不是其他类型参数的替换),而函数的参数的传递,参数是有数据类型的,可以是各种各样的类型.宏的参数替换是不经计算而直接处理的,而函数调用是将实参的值传递给形参,既然说是值,自然是计算得来的.宏在编译之前进行,即先用宏体替换宏名,然后再编译的,而函数显然是编译之后,在执行时,才调用的.因此,宏占用的是编译的时间,而函数占用的是执行时的时间.宏的参数是不占内存空间的,因为只是做字符串的替换,而函数调用时的参数传递则是具体变量之间的信息传递,形参作为函数的局部变量,显然是占用内存的.函数的调用是需要付出一定的时空开销的,因为系统在调用函数时,要保留现场,然后转入被调用函数去执行,调用完,再返回主调函数,此时再恢复现场,这些操作,显然在宏中是没有的.宏替换不占用程序的运行时间宏与类型无关但是c中函数必须指定返回类型故宏可以做函数不能做的事六函数模板1.函数模板的格式如下Template class 形参名class 形参名...... 返回类型函数名(参数列表){函数体}其中class可以用typename关键字代替2.函数模板调用时不需要显式指定类型系统自动匹配参数类型若没有合适的会进行报错。而类模板使用需要显式指定类型,且对于函数模板注意要返回值和参数的类型一致3.模板函数和普通函数都符合条件时优先执行普通函数4.模板特化当函数模板需要对某些类型进行特化处理称为函数模板的特化类模板的特化同理① 因为很多时候我们既需要一个模板能应对各种情形又需要它对于某个特定的类型有着特别的处理故出现了模板特化① 特化整体上分为全特化和偏特化② 全特化:就是模板中模板参数全被指定为确定的类型。 全特化也就是定义了一个全新的类型全特化的类中的函数可以与模板类不一样③ 偏特化模板中的模板参数没有被全部确定需要编译器在编译时进行确定④ 对主版本模板类、全特化类、偏特化类的调用优先级从高到低进行排序是全特化类偏特化类主版本模板类⑤ 当函数调用发现有特化后的匹配函数时会优先调用特化的函数而不再通过函数模版来进行实例化⑥ 模板特化相当于在声明了类模板等后声明需要模板特化然后让接下来的代码自己使用一个类型故不能单独使用例![img](file:///C:\Users\ADMINI~1\AppData\Local\Temp\ksohtml\wps386C.tmp.jpg)5. 模板特化实现template 完全和模板类型没有一点关系的类实现或者函数定义① 特化为绝对类型全特化:例:templateclass Compare{...}; // 特化为float类型此为函数模板特化函数模板只能全特化没有偏特化② 特化为引用指针类型(半特化、偏特化)例template struct iterator_traits_Tp* {}③ 特化为另外一个类模板偏特化例template class Comparevector{}6. 类模板的成员函数都是函数模板没使用过的成员函数即函数模板不会被实例化7. 函数模板必须由程序员实例化为可执行的函数8. 函数模板的虚拟类型名是在编译阶段确定实际类型的七函数重载使用重载函数编程序的目的是使用相同的函数名调用功能相似的函数使用方便提高可读性重载函数的形参个数或类型必须不同void x(int,char ch’a’)与void x(int)可以在同一程序中定义但不可以重载八内联函数内联置函数inline引入内联函数的目的是为了解决程序中函数调用的效率问题程序在编译器编译的时候编译器将程序中出现的内联函数的调用表达式用内联函数的函数体进行替换而对于其他的函数都是在运行时候才被替代。这其实就是个空间代价换时间的节省弊代码被多次复制增加了代码量占用更多的内存空间故在当函数代码较小并且被频繁调用的时候。在使用内联函数时要留神① 使用基类指针或引用来调用虚函数时它都不能为内联函数(因为调用发生在运行时)。但是使用类的对象(不是指针或引用)来调用时可以当做是内联因为编译器在编译时确切知道对象是哪个类的② 默认情况下在类体中定义的成员函数若不包括循环等控制结构符合内联函数要求时C会自动将它们作为内联函数处理不是所有成员函数都是内联函数③ 内联函数在编译时是将该函数的目标代码插入每个调用该函数的地方不是运行时④ 内联函数在编译时做参数类型检查⑤ 在内联函数中不允许使用循环语句for,while和switch结果带有异常接口声明的函数也不能声明为内联函数。另外递归函数(自己调用自己的函数)是不能被用来做内联函数的。内联函数只适合于只有15行的小函数⑥ 内联函数的定义必须出现在内联函数第一次调用之前⑦ 定义内联函数inline写类型前面⑧ 关键字inline 必须与函数定义体放在一起才能使函数成为内联仅将inline 放在函数声明前面不起任何作用⑨ 如果在类外定义inline函数则必须将类定义和成员函数定义放在同一头文件中否则编译时无法进行置换⑩ 头文件中不仅要包含 inline 函数的声明而且必须包含定义且在定义时必须加上 inline⑪ 不管是 class 声明中定义的 inline 函数还是 class 实现中定义的 inline 函数不存在优先不优先的问题⑫ 内置函数不需要使用堆栈进行现场的保护与恢复⑬ 用 inline 修饰的函数原型其对应的函数也将成为内联函数 - 错自己理解inline为建议型关键字⑭ 内联函数可以是静态的