前言:
本文是我在重学 C++ 时所记录的学习过程,笔记中未记录较为简单的语法知识,但却总结了许多容易忘记的薄弱点,如指针应用、别名、时钟应用等,教材为 C++ Primer Plus 。
〇、零碎知识
一、IO函数
二、复合类型
三、表达式
四、函数
五、内存模型和名称空间
六、类与对象
七、类的高级特性
八、代码重用
九、友元、异常和其他
十、标准模板库
十一、输入、输出和文件
十二、探讨 C++ 新标准
习题
1. C++程序的模块叫什么?
答:函数。
2. #include<iostream>
预处理器编译指令是做什么用的?
答:在最终编译前,使用iostream
文件中的内容替换该编译指令。
3. using namespace std
是做什么用的?
答:使程序可以使用std
名称空间中的定义。
4. C++提供了什么措施来防止超出整型的范围?
答:C++没有提供自动防止超出整型限制的功能,可以用头文件climits
来确定限制情况。
5.入口条件循环和出口条件循环之间的区别是什么?各种C++循环分别属于其中的哪一种?
答:输入条件循环在进入输入循环体之前将评估测试表达式,如果条件最初为false
,则不会执行循环体。退出条件循环在处理循环体之后评估测试表达式。因此for、while
属于输入条件循环,do-while
属于退出条件循环。
6. 在查看输入方面,cin >> ch
同cin.get(ch)
和ch = cin.get()
有什么不同?
答:cin >> ch
将跳过空格、换行符和制表符,其他两种格式将读取这些字符。
7. 使用函数的三个步骤是什么?
答:定义函数、提供原型、调用函数
8. 为什么不对 类型 为 基本类型 的 函数参数 使用 const 限定符?
答:将 const 限定符用于指针,以防止指向的原始数据被修改。程序传递基本类型时,它将按值传递,以便函数使用副本。这样保护了原始数据。
9. 哪种函数适合定义为内联函数?
答:只有一行代码的小型、非递归函数。
10. 什么是类?
答:类是用户定义的类型的定义。类声明指定了数据将如何存储,同时指定了用来访问和操纵这些数据的方法(类成员函数)。
11. 类如何实现抽象、封装和数据隐藏?
答:
- 抽象:类表示人们可以用类方法的公有接口对类对象执行的操作。
- 数据隐藏:类的数据成员可以是私有的(默认值),这意味着只能通过成员函数来访问这些数据。
- 封装:实现的具体细节(如数据表示和方法的代码)都是隐藏的。
12. 对象和类之间的关系是什么?
答:类定义了一种类型,包括如何使用它。对象是一个变量或其他数据对象(如由new
生成的),并根据类定义被创建和使用。类和对象之间的关系同标准类型与其变量之间的关系相同。
13. 除了是函数之外,类函数成员与类数据成员之间的区别是什么?
答:如果创建给定类的多个对象,则每个对象都有其自己的数据内存空间;但所有的对象都使用同一组成员函数(通常方法是公有的,而数据是私有的)。
14. 友元函数与成员函数之间的区别是什么?
答:成员函数是类定义的一部分,通过特定的对象来调用。成员函数可以隐式访问调用对象的成员,而无需使用成员运算符。友元函数不是类的组成部分,因此被称为直接函数调用。友元函数不能直接访问类成员,而必须将成员运算符用于作用参数传递的对象。
15. 非友元函数必须是友元才能访问类成员吗?
答:要访问私有成员,它必须是友元,但要访问公有成员,可以不是友元。
16. 派生类从基类那里继承了什么?
答:基类的公有成员成为派生类的公有成员;基类的保护成员成为派生类的保护成员;基类的私有成员被继承,但不能直接访问。
17. 派生类不能从基类那里继承什么?
答:不能继承构造函数、析构函数、赋值运算符和友元。
18. 创建和删除派生类对象时,构造函数和析构函数调用的顺序是怎样的?
答:按派生的顺序调用构造函数,最早的构造函数最先调用。调用析构函数的顺序刚好相反。
19. 如果派生类没有添加任何数据成员,它是否需要构造函数?
答:需要,每个类都必须有自己的构造函数。
20. 如果基类和派生类定义了同名的方法,当派生类对象调用该方法时,被调用的将是那个方法?
答:只调用派生类方法。它取代基类定义。仅当派生类没有重新定义方法或使用作用域解析运算符时,才会调用基类方法。然而,应把所有要重新定义的函数定义为虚函数。
21. 在什么情况下派生类要定义赋值运算符?
答:如果派生类构造函数使用new
或new[]
运算符来初始化类的指针成员,则应定义一个赋值运算符。更普遍的说,如果对于派生类成员来说,默认赋值不正确,则应定义赋值运算符。
22. 可以将派生类对象的地址赋给基类指针吗?可以将基类对象的地址赋给派生类指针吗?
答:当然,可以将派生类对象的地址赋给基类指针;但只有通过显式类型转换,才可以将基类对象的地址赋给派生类指针(向下转换),而使用这样的指针不一定安全。
23. 可以将派生类对象赋给基类对象吗?可以将基类对象赋给派生类对象吗?
答:是的,可以将派生类对象赋给基类对象。对于派生类中新增的数据成员都不会传递给基类对象。然而,程序将使用基类的赋值运算符。仅当派生类定义了转换运算符(即包含将基类引用作为唯一参数的构造函数)或使用基类为参数的赋值运算符时,相反方向的赋值才是可能的。
24. 假设定义了一个函数,它将基类对象的引用作为参数。为什么该函数也可以将派生类对象作为参数?
答:它可以这样做,因为C++允许基类引用指向从该基类派生而来的任何类型。
25. 假设定义了-一个函数,它将基类对象作为参数(即函数按值传递基类对象)。为什么该函数也可以将派生类对象作为参数?
答:按值传递对象将调用复制构造函数。由于形参是基类对象,因此将调用基类的复制构造函数。复制构造函数以基类引用为参数,该引用可以指向作为参数传递的派生对象。最终结果是,将生成一个新的基类对象,其成员对应于派生对象的基类部分。
26. 为什么通常按引用传递对象比按值传递对象的效率更高?
答:按引用(而不是按值)传递对象,这样可以确保函数从虚函数受益。另外,按引用(而不是按值)传递对象可以节省内存和时间,尤其对于大型对象。按值传递对象的主要优点在于可以保护原始数据,但可以通过将引用作为const
类型传递,来达到同样的目的。基类对象,其成员对应于派生对象的基类部分。
27. iostream
文件在C++ I/O中扮演何种角色?
答:iostream
文件定义了用于管理输入和输出的类、常量和操纵符,这些对象管理用于处理 I/O 的流和缓冲区。该文件还创建了一些标准对象(cin
、cout
、cerr
和clog
以及对应的宽字符对象),用于处理与每个程序相连的标准输入和输出流。
28. 为什么键入数字(如121)作为输入要求程序进行转换?
答:键盘输入生成一系列字符。输入 121 将生成 3 个字符,每个字符都由一个 1 字节的二进制码表示。要将这个值存储为int
类型,则必须将这 3 个字符转换为 121 值的二进制表示。
29. 标准输出与标准错误之间有什么区别?
答:在默认情况下,标准输出和标准错误都将输出发送给标准输出设备(通常为显示器)。然而,如果要求操作系统将输出重定向到文件,则标准输出将与文件(而不是显示器)相连,但标准错误仍与显示器相连。
30. 输入方法的定义的哪一个特征能够拼接输出?
答:可以返回ostream &
类型的输出方法拼接。这样,通过一个对象调用方法时,将返回该对象。然后,返回对象将可以调用序列中的下一个方法。