第六章C++:函数基础与应用

第六章 函数

函数是一个命名了的代码块,通过调用函数执行相应的代码。

函数基础

通过调用运算符(call operator)来执行函数。其形式是一对圆括号。

函数的调用完成两项工作(如下),此时主调函数(calling function)的执行被暂时中断,被调函数(called function)开始执行。

用实参初始化函数对应的形参。

将控制权转移给被调函数。

立即学习“C++免费学习笔记(深入)”;

return语句:

返回return语句中的值

将控制权从被调函数移回主调函数

局部对象

名字有作用域,对象有生命周期(lifetime)

自动对象(automatic object):当函数的控制路径经过变量定义语句时创建该对象,当达到定义所在的块末尾时销毁它。

局部静态对象:程序执行路径第一次经过对象定义语句时初始化,知道程序终止才被销毁。

将局部变量定义为static获得,例如:

//统计函数count_calls ()被调用了多少次
size_t count_calls ()
{
static size_t ctr = 0;  //调用结束后,这个值仍然有效
return ++ctr;
}
int main()
{
for (size_t i = 0; i != 10; ++i)
cout return 0;
}

函数声明

也称作函数原型(function prototype)

函数三要素(返回类型、函数名、形参类型)描述了函数的接口,函数声明中形参名可省略。

函数应在头文件中声明,源文件中定义。

分离式编译

参数传递

如果形参是引用类型,它将绑定到对应的实参上;否则,将实参的值拷贝后赋给形参。
– 如果无需修改引用形参的值,最好将其声明为常量引用。

main:处理命令行选项

假设main函数位于可执行文件prog内,我们可以向程序传递下面的选项:

prog -d -o ofile data0

这些命令通过两个可选的形参传递给main函数:

int main(int argc, char *argv[]) {...}//或:int main(int argc, char **argv) {...}

当实参传给main函数之后,argv的第一个元素指向程序的名字或者一个空字符串,接下来的元素一次传递命令行提供的实参。最后一个指针只会掉元素值保证为0。
– 以上面的命令行为例:

argc = 5;argv[0] = "prog";argv[1] = "-d";argv[2] = "-o";argv[3] = "ofile";argv[4] = "data0";argv[5] = 0;

含有可变形参的函数

C++11新标准提供两种方法编写能处理不同数量实参的函数:

所有实参类型相同,可以传递一个名为initializer_list的标准库类型。

实参类型不同,我们可以编写一种特殊的函数,叫做可变参数模板。

C++还有一种特殊的形参类型:省略符。可以用它传递可变数量的实参。这种功能一般只用于与C函数交互的接口程序。

initializer_list形参

其类型定义在同名的头文件中

提供如下操作:

initializer_list lst;    //默认初始化,T类型元素的空列表
initializer_list lst{a,b,c...};
//lst的元素数量和初始值一样多;lst的元素是对应初始值的副本;列表中的元素是const
lst2(lst)
lst2 = lst  //拷贝或复制一个initializer_list对象不会拷贝列表中的元素;拷贝后,原始列表和副本元素共享
lst.size()  //列表中元素的数量
lst.begin() //返回指向lst中首元素的指针
lst.end()   //返回指向lst中尾元素下一位置的指针

返回类型和return语句

引用返回左值,其他返回类型得到右值。

列表初始化返回值:C++11新标准规定,函数可以返回花括号包围的值的列表。

主函数main的返回值

允许main函数没有返回值(若没有,编译器隐式地插入return 0)

返回0表示执行成功,其他值依机器而定。

为了使返回值与机器无关,cstdlib头文件定义了两个预处理变量,分别表示成功和失败:

return EXIT_FAILURE;
return EXIT_SUCCESS;
//因为它们是预处理变量,所以既不能在前面加上std::,也不能在using声明里出现。

返回数组指针

使用类型别名

typedef int arrT[10];   //arrT是一个类型别名,它表示的类型是含有10个整数的数组
using arrT = int[10];   //与上一句等价
arrT* func(int i);      //func返回一个指向含有10个整数的数组的指针

声明一个返回数组指针的函数,形式如下

Type (*function(parameter_list)) [dimension]
//Type表示返回的数组指针指向的数组元素类型
//dimension表示数组的大小
//例如:
int (*func(int i)) [10];

使用尾置返回类型(C++11)

auto func(int i) -> int(*)[10];

使用decltype

int odd[] = {1,3,5,7,9};
int even[] = {0,2,4,6,8};
decltype(odd) *arrPtr(int i)
{
   return (i % 2) ? &odd : &even;  //返回一个指向数组的指针
}

函数重载

如果同一作用域内的几个函数名字相同但形参列表不同,我们称之为重载(overloaded)函数

不允许两个函数除了返回类型外其他所有要素都相同。

重载与作用域:一旦在当前作用域中找到了所需的名字,编译器就会忽略掉外层作用域中的同名实体。

特殊用途语言特性

介绍三种函数相关的语言特性:默认实参、内联函数、constexpr函数。

默认实参

调用包含默认实参的函数时,可以包含该实参,也可以省略该实参。

一旦某个形参被赋予了默认值,它后面所有的形参都必须有默认值。

内联函数(inline)

调用函数一般比求等价表达式的值要慢,内联函数可避免函数调用的开销。
– 将函数指定为内联函数,通常就是将它在每个调用点上“内联地”展开。

constexpr函数

函数的返回类型和所有的形参类型都得是字面值类型。

函数中必须有且只有一条return语句。

constexpr函数被隐式地指定为内联函数。

内联函数和constexpr函数通常定义在头文件中

调试帮助

程序可以包含一些用于调试的代码,但这些代码只在开发程序时使用。当应用程序编写完成准备发布时,要先屏蔽掉调试代码。这种方法用到两项预处理功能:assert和NDEBUG。

assert预处理宏

#include assert(expr);//首先对expr求值,//如果表达式为假(即0),assert输出信息并终止程序的执行。//如果表达式为真(即非0),assert什么也不做。//例如:对一个文本进行操作的程序可能要求所给定单词的长度都大于某个阈值。assert(word.size() > threshold;

NDEBUG预处理变量

assert的行为依赖于一个名为NDEBUG的预处理变量的状态。如果定义了NDEBUG,则assert什么也不做。默认状态下没有定义NDEBUG,此时assert将运行执行时检查。

使用#define语句定义NDEBUG,从而关闭调试状态。

很多编译器都提供了命令行选项使我们可以定义预处理变量。

$ CC -D NDEBUG main.C   #微软编译器中用 /D

这只是调试程序的辅助手段,不能代替真正的逻辑检查,也不能代替程序本应该包含的错误检查。

除了assert以外,也能使用NDEBUG编写自己的条件调试代码:

//如果定义了NDEBUG,#ifndef和#endif之间的代码将被忽略void print(const int ia[], aize_t size){    #ifndef NDEBUG        //_ _func_ _是编译器定义的一个局部静态变量,用于存放函数的名字,它是const char的一个静态数组。        cerr << _ _func_ _ << "array size is " << size << endl;    #endif}

除了_ _ func _ _之外,还有其它四个名字:

_ _FILE_ _ 存放文件名的字符串字面值_ _LINE_ _ 存放当前行号的整型字面值_ _TIME_ _ 存放文件编译时间的字符串字面值_ _DATA_ _ 存放文件编译日期的字符串字面值

函数指针

bool lengthCompare(const string &, const string &);//pf指向一个函数,该函数的参数是两个const string的引用,返回值是bool类型。注意圆括号必不可少bool (*pf) (const string &, const string &);    //未初始化

当我们把函数名作为值使用时,该函数自动地转换成指针

pf = lengthCompare;     //pf指向名为lengthCompare的函数pf = &lengthCompare;    //等价赋值语句,&是可选的

调用该函数:

//此三个调用等价bool b1 = pf("hello", "goodbye");bool b2 = (*pf)("hello", "goodbye");bool b3 = lengthCompare("hello", "goodbye");

参考:C++Primer第五版

相关文章:

第四章C++:表达式概念-运算符的应用

第五章C++:语句的相关介绍

以上就是第六章C++:函数基础与应用的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432872.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 08:41:22
下一篇 2025年12月12日 19:03:22

相关推荐

  • 区分C++常量表达式、const、constexpr(附代码)

    常量表达式是指值不会改变且在编译过程中就能够得到计算结果的表达式,能在编译时求值的表达式。 例1: #include using namespace std;int main(){const int a1 = 10; // a1是常量表达式。const int a2 = a1 + 20; // a2…

    2025年12月17日
    000
  • C#学习记录:编写高质量代码改善整理建议1-3

    建议1:正确使用字符串 string  string str1 = “str1” + 9;string str2 = “str2” + 9.ToString(); 第一行代码会产生一次装箱,还有一次string的concat 而第二行代码使用ToString(),内部使用的是Number.Forma…

    2025年12月17日 好文分享
    000
  • C#学习记录:编写高质量代码改善整理建议4-8

    4、TryParse比Parse好 下面的tryparse的方法定义 public static bool TryParse(string s, out Double result); Parse如果转换失败会报错,但是TryParse有返回值可以判断是否转换成功 string str1 = “ab…

    2025年12月17日
    000
  • C#学习记录:编写高质量代码改善整理建议9-15

    9、习惯重载运算符 在构建自己的类型时,始终应该考虑是否可以使用运算符重载 10、创建对象时需要考虑是否实现比较器 如果需要排序,有两种比较器实现 class FirstType : IComparable{ public string name; public int age; public Fi…

    2025年12月17日 好文分享
    000
  • C++类的静态数据成员和静态成员函数

    静态数据成员 ·用关键字static声明 ·当声明类的数据成员为静态时,无论创建多少个类的对象,静态成员都只有一个副本 ·在类的所有对象中共享,具有静态生存期 ·若不存在其他的初始化语句,在创建第一个对象时,所有的静态数据成员被初始化为零 立即学习“C++免费学习笔记(深入)”; ·在类外定义和初始…

    好文分享 2025年12月17日
    000
  • C# 使用NPOI生成Word文档(按照模板)

    项目需要集成word导出,做的时候网上文档资源不是很多,也比较杂乱,所以查了查,整理了整理,做个记录,也顺便把NPOI操作Word文档的一些基本操作分享给有需要的朋友。   本篇包括生成word对word文本的操作,表格的操作,以及图片的操作,都为生成word基础的一些操作。 以下只是我的个人理解所…

    好文分享 2025年12月17日
    000
  • C#入门经典学习阶段小结(凌乱)

    CLR: common language runtime(.net 公共语言运行库)管理.net库开发的所有应用程序的执行 cts: common type system (通用类型系统)指定最基本类型有助于使用.net framework的各种语言之间进行交互操作cil:common interm…

    好文分享 2025年12月17日
    000
  • 从事C/C++开发多年,给零基础想学习C/C++的几点建议

    我在C++领域也从事了好多年了,也是有一些经验心得的,想分享给那些零基础入门C语言的伙伴们。从C++入门到精通各需要学习什么以及注意事项,避免不必要的麻烦,也很浪费时间,经验交流,如有不足之处多多谅解,我也只是简单的介绍。apache php mysql 从事C/C++开发多年,给零基础想学习C/C…

    2025年12月17日
    000
  • 案例分享c++ map的使用和 查找性能测试

    最近在为推荐服务作性能调优,这个服务的主要逻辑是用离线计算的模型数据给请求中的每个广告打分,再返回这些广告的排序结果,这里面打分的过程其实就用请求中的数据拼成各种key,去查一个大的 map,这种计算非常多,成为了主要的性能瓶颈,代码比较老,使用的是 boost::unordered_map,为了解…

    好文分享 2025年12月17日
    000
  • 基于汇编的 C/C++ 协程(用于服务器)的实现

    本篇文章,是 对c++/c++ 协程的实现。我们需要实现这两个目标: 有同步式服务器编程的顺序思路,便于功能设计和代码调试——我使用了 libco 中的协程部分 有异步 I/O 的性能——我使用了 libevent 中的 event I/O     apache php mysql 结构上,就是将 …

    2025年12月17日 好文分享
    000
  • C++ 布尔类型和引用的用法详解

    1、布尔类型 c++kquote>C++中的布尔类型C++在C语言的基本类型系统之上增加了boolC++中的bool可取的值只有true和false理论上bool之占用一个字节注意:true代表真值,编译器内部用1来表示false代表非真值, 编译器内部用0来表示C语言中:用整型值来代替boo…

    2025年12月17日
    000
  • C++引用的意义与引用的本质

    1、引用的意义 引用作为变量别名而存在,因此在一些场合可以替代指针,引用相对于指针来说具有更好的可读性和实用性 // swap函数的实现对比void swap(int& a, int& b){ int t = a; a = b; b = t;}void swap(int* a, in…

    2025年12月17日
    000
  • 第一章C++:函数返回值、GNU编译器命令

    c++bce3b83f770dfdf50c5dae0e4360a>函数返回值 返回值类型必须和函数的返回类型相同 对于main函数,返回值必须是int型。大多数系统中,main函数的返回值是一个状态指示器。返回值为0表示main函数成功执行完毕;返回非零表示有错误出现。 GNU编译器命令 vi…

    好文分享 2025年12月17日
    000
  • 第二章C++:变量和基本类型

    c++bce3b83f770dfdf50c5dae0e4360a>介绍语言的基本知识和标准库 第二章 变量和基本类型 一些语言,如smalltalk和python等,在程序运行时检查数据类型;与之相反,c++是一种静态数据类型语言,它的类型检查发生在编译时。 – 内置类型(buil…

    好文分享 2025年12月17日
    000
  • 第三章C++:字符串string、向量vector和数组

    第二章介绍内置类型,本章介绍抽象数据类型库。  – 其中string和vector是两种最重要的标准库类型,前者支持可变长字符串,后者表示可变长的集合。 – 还有一种标准库类型是迭代器,它是string和vector的配套类型,常被用于访问string中的字符和vector中…

    好文分享 2025年12月17日
    000
  • 利用Unity脚本自定义分辨率实现相机截一张高清截图

    最近做项目的时候需要在游戏里截一张高清截图,研究了一下写成脚本,方便以后使用。脚本可以自定义分辨率,用相机截高清截图。可以用代码动态截图,也可以在编辑模式下截图。注意截图宽高比要正确,宽高比不正确时可能会出问题。 截图效果: 脚本:CameraCapture.cs using UnityEngine…

    2025年12月17日
    000
  • 技术解答CSV 文件的一个 .NET 库:CsvHelper 中文文档

    csvhelper 是读写 csv 文件的一个 .net 库。可以通过 visual studio 的包管理器下载 csvhelper。自动映射定义:没有提供映射文件的情况下,默认为自动映射,自动映射会按顺序依次映射到类的属性中去。 GitHub 地址 读取 读取所有的记录 var csv = ne…

    好文分享 2025年12月17日
    000
  • 简要分析Unity计时器脚本Timer的用法(附代码)

    计时器效果图: Timer用法:第一种:脚本加到物体上,勾选”自动计时”。第二种:脚本加到物体上,调用timer.start()方法启动。第三种:代码中动态添加Timer脚本。 using UnityEngine;public class TimerTest : MonoBe…

    2025年12月17日
    000
  • 常见的C++中const常量用法分析讲解

    先对c++语言中的const和c++中的const进行讲解,前者const修饰的变量不是真的常量,它只是告诉编译器该变量不能出现在赋值符号的左边。后者c++在c的基础上对const进行了进化处理。 1、C语言中的const: const修饰的变量是只读的,本质还是变量 const修饰的局部变量在栈上…

    好文分享 2025年12月17日
    000
  • 深入理解C#设计模式之策略模式 角色具体案例分享

    策略模式(Stragety Pattern) 策略模式属于行为型模式,它定义了一系列的算法,并将每一个算法封装起来,而且使他们可以相互替换,让算法独立于使用它的客户而独立变化。 使用策略模式可以把行为和环境分割开来。环境类负责维持和查询行为类,各种算法则在具体策略类中提供。 角色: 1、抽象策略(S…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信