一、单项选择题(本大题共20小题,每小题1分,共20分)在每小题列出的四个备选项中 只有一个是符合题目要求的,请将其代码填写在题后的括号内。错选、多选或未选均无 分。
1. 编写C++程序一般需经过的几个步骤依次是() A. 编辑、调试、编译、连接 B. 编辑、编译、连接、运行 C. 编译、调试、编辑、连接 D. 编译、编辑、连接、运行 答案:B
解析:经过编辑、编译、连接和运行四个步骤。编辑是将C++源程序输入计算机的过程,保 存文件名为cpp。编译是使用系统提供的编译器将源程序cpp生成机器语言的过程,目标文件为
obj,由于没有得到系统分配的绝对地址,还不能直接运行。连接是将目标文件obj转换为可执行
程序的过程,结果为exe。运行是执行exe,在屏幕上显示结果的过程。 2. 决定C++语言中函数的返回值类型的是() A. return语句中的表达式类型
B. 调用该函数时系统随机产生的类型 C. 调用该函数时的主调用函数类型 D. 在定义该函数时所指定的数据类型 答案:D
解析:函数的返回值类型由定义函数时的指定的数据类型决定的。A项的表达式的值要转换 成函数的定义时的返回类型。 3. 下面叙述不正确的是() A. 派生类一般都用公有派生
B. 对基类成员的访问必须是无二义性的 C. 赋值兼容规则也适用于多重继承的组合 D. 基类的公有成员在派生类中仍然是公有的 答案:D
解析:继承方式有三种:公有、私有和保护。多继承中,多个基类具有同名成员,在它们 的子类中访问这些成员,就产生了二义性,但进行访问时,不能存在二义性。赋值兼容规则是指
派生类对象可以当作基类对象使用,只要存在继承关系,所以单继承或多继承都适用。基类中的
公有成员采用私有继承时,在派生类中变成了私有成员,所以D项错误。
4. 所谓数据封装就是将一组数据和与这组数据有关操作组装在一起,形成一个实体,这实体
也就是() A. 类 B. 对象 C. 函数体 D. 数据块 答案:A
解析:类即数据和操作的组合体,数据是类的静态特征,操作是类具有的动作。
5. 在公有派生类的成员函数不能直接访问基类中继承来的某个成员,则该成员一定是基类中 的()
A. 私有成员 B. 公有成员 C. 保护成员
D. 保护成员或私有成员 答案:A
解析:在派生类中基类的保护或者基类公有都可以直接访问,基类的私有成员只能是基类 的成员函数来访问。所以选择A项。
6. 对基类和派生类的关系描述中,错误的是() A. 派生类是基类的具体化 B. 基类继承了派生类的属性 C. 派生类是基类定义的延续 D. 派生类是基类的特殊化 答案:B
解析:派生类的成员一个是来自基类,一个来自本身,所以派生类是基类的扩展,也是基 类的具体化和特殊化,派生类是对基类扩展。B项基类不能继承派生类成员,所以错误。 7. 关于this指针使用说法正确的是()
A. 保证每个对象拥有自己的数据成员,但共享处理这些数据的代码 B. 保证基类私有成员在子类中可以被访问。 C. 保证基类保护成员在子类中可以被访问。 D. 保证基类公有成员在子类中可以被访问。 答案:A
解析:this指针是隐藏的,可以使用该指针来访问调用对象中的数据。基类的成员在派生 类中能否访问,与继承方式有关,与this没有关系。所以选择A项。 8. 所谓多态性是指()
A. 不同的对象调用不同名称的函数 B. 不同的对象调用相同名称的函数 C. 一个对象调用不同名称的函数 D. 一个对象调用不同名称的对象 答案:B
解析:多态性有两种静态多态性和动态多态性,静态多态性是指调用同名函数,由于参数 的不同调用不同的同名函数;动态多态性是指不同对象调用同名函数时,由于对象不同调用不同
的同名函数。多态性肯定具有相同的函数名,所以选择B项。
9. 一个函数功能不太复杂,但要求被频繁调用,则应把它定义为() A. 内联函数 B. 重载函数 C. 递归函数 D. 嵌套函数 答案:A
解析:内联函数特征代码少,频繁调用,执行效率高。重载函数解决统一接口的问题;递 归是子程序调用,程序调用要耗费很多空间和时间,循环/迭代都比递归有效率得多,递归
只是
从形式上,逻辑比较简洁。嵌套函数即反复调用,速度较慢。所以选择A项。 10. 下面函数模板定义中不正确的是() A. A B. B C. C D. D 答案:A
解析:A项中F是一个返回Q类型的值,而return中用返回类型作为返回值错误。所以选择 A项。
11. 假设ClassY:publicX,即类Y是类X的派生类,则说明一个Y类的对象时和删除Y类对象时
,调用构造函数和析构函数的次序分别为() A. X,Y;Y,X B. X,Y;X,Y C. Y,X;X,Y D. Y,X;Y,X 答案:A
解析:派生类构造函数必须对这三类成员进行初始化,其执行顺序:调用基类构造函数 ;调用子对象的构造函数;派生类的构造函数体。析构函数在执行过程中也要对基类和成员对象
进行操作,但它的执行过程与构造函数正好相反,即对派生类新增普通成员进行清理;调用成员
对象析构函数,对派生类新增的成员对象进行清理;调用基类析构函数,对基类进行清理,所以
选择A项。
12. 适宜采用inline定义函数情况是() A. 函数体含有循环语句 B. 函数体含有递归语句 C. 函数代码少、频繁调用 D. 函数代码多、不常调用 答案:C
解析:内联函数具有程序代码少、频繁调用和执行效率高的特征,所以选择C项。 13. 假定一个类的构造函数为A(int aa,int bb) {a=aa--;b=a*bb;},则执行A x(4,5);语句后 ,x.a和x.b的值分别为() A. 3和15 B. 5和4 C. 4和20 D. 20和5 答案:C
解析:a=4,因为后减,b的值与a、bb相关,b=4*5=20,而与aa没有任何关系。 14. 在类中说明的成员可以使用关键字的是() A. public B. extern
C. cpu D. register 答案:A
解析:extern用于声明外部变量的。register声明寄存器类型变量。无cpu类型。它们都不能声
明类成员。public声明为公有访问权限,所以选择A项。 15. 下列不能作为类的成员的是() A. 自身类对象的指针 B. 自身类对象
C. 自身类对象的引用 D. 另一个类的对象 答案:B 解析:类的定义,如果有自身类对象,使得循环定义,B项错误。在类中具有自身类的指针,可
以实现链表的操作,当然也可以使用对象的引用。类中可以有另一个类的对象,即成员对象。所
以选择B选项。
16. 使用地址作为实参传给形参,下列说法正确的是() A. 实参是形参的备份 B. 实参与形参无联系 C. 形参是实参的备份
D. 实参与形参是同一对象 答案:D
解析:地址作为实参,表示实参与形参代表同一个对象。如果实参是数值,形参也是普通 变量,此时形参是实参的备份。所以选择D项。
17. 下列程序的输出结果是() #include
cout<
解析:如果数组元素都是相同类型的指针,则称这个数组为指针数组。指针数组一般用于处理二
维数组。声明的格式为:<数据类型><(*变量名)><[元素个数]>。
p表示指向数组n的行指针。如果将指针的初始化(*p)[3]=b;地址的等价形式: p+i p[i]*(p+i)都表示b数组第i+1行的第1个元素的首地址。 *(p+i)+jp[i]+j &p[i][j]都表示b数组第i+1行、第j+1列元素的地址。
值的等价形式:
*(*(p+i)+j) *(p[i]+j) p[i][j]都表示b数组第i+1、第j+1列元素的值。 所以题目分别访问p[0][0],p[0][1],p[0][2]。 18. 在C++中,使用流进行输入输出,其中用于屏幕输入() A. cin B. cerr C. cout D. clog
答案:A
解析:(1)标准输入流cin:istream类的对象。(2)标准输出流cout:ostream类的对象。 (3)非缓冲型标准出错流cerr:ostream类的对象。(4)缓冲型标准出错流clog:ostream类的对象
19. 假定AA为一个类,a()为该类公有的函数成员,x为该类的一个对象,则访问x对象中函数
成员a()的格式为() A. x.a B. x.a() C. x->a
D. (*x).a() 答案:B
解析:对象访问成员的方式为:对象名.成员。指针可以有两种:(*对象指针).成员或者对 象指针->成员。A选项是访问数据成员,B项是访问成员函数。 20. 关于对象概念的描述中,说法错误的是() A. 对象就是C语言中的结构变量
B. 对象代表着正在创建的系统中的一个实体 C. 对象是类的一个变量
D. 对象之间的信息传递是通过消息进行的 答案:A
解析:A对象在C++中才有,包括数据和操作两项,而C中的变量只有数据,没有操作。所
以A项错误。
二、填空题(本大题共20小题,每小题1分,共20分)请在每小题的空格中填上正确答案 。错填、不填均无分。
1. C++的流库预定义了4个流,它们是cin、cout、clog和___。 答案:(P193)cerr
[解析]cin、cout、clog和cerr分别用于标准输入、输出、标准错误流(缓冲)和标准错误流
(非缓冲)。
2. 每个对象都是所属类的一个___。 答案:实例
[解析]类是对象的抽象,对象是类的一个实例。
3. 在已经定义了整型指针ip后,为了得到一个包括10个整数的数组并由ip所指向,应使用语 句___。
答案:int *ip=new int[10];
[解析]new用来动态开辟空间。常用来产生动态数组及对象构造函数。 4. 函数模板中紧随template之后尖括号内的类型参数都要冠以保留字___。 答案:class
[解析]类模板的使用。template
5. 定义类的动态对象数组时,系统只能够自动调用该类的___构造函数对其进行初始化。 答案:无参
[解析]使用new创建对象数组,调用无参构造函数。 6. 表达式cout< [解析]endl与字符常量‘\n’等价。 7. 在C++中,访问一个指针所指向的对象的成员所用的指向运算符是___。 答案:-> [解析]指针使用成员有两种方法:“->”指向运算符和“.”成员访问运算符。 8. 假如一个类的名称为MyClass,使用这个类的一个对象初始化该类的另一个对象时,可以调 用___构造函数来完成此功能。 答案:复制或拷贝 复制或拷贝构造函数就是用对象初始化新的对象。 9. 对赋值运算符进行重载时,应声明为___函数。 答案:(P183)类成员 [解析]运算符重载的方法有友元或者成员函数两种途径,但是赋值运算符只能使用成员函数的 方法来实现。 10. 如果要把A类成员函数f()且返回值为void声明为类B的友元函数,则应在类B的定义中加 入的语句___。 答案:(P109)friend void A::f(); [解析]成员函数作为另一个类的友元函数,格式为:friend 返回类型类名::函数(形参)。 11. 下列程序段的输出结果是___。 for(i=0,j=10,k=0;i<=j;i++,j-=3,k=i+j);cout< 答案:4 [解析]for循环结构,三个表达式的作用,初始化、循环判断条件和循环变量变化。循环执行了 三次,k的作用是计算i、j的和。 12. String 类的___方法返回查找到的字符串在主串的位置。 答案:(P40)find [解析]string类对象方法的find,查不到字符串,则返回-1。 13. int n=0; while(n=1)n++; while循环执行次数是___。 答案:无限次 [解析]=是赋值运算符,不是关系运算符,且不等0,所以死循环。 14. 控制格式输入输出的操作中,函数___是用来设置填充字符。要求给出函数名和参数类型 答案:setfill(char) [解析]格式控制方法的使用,如setw,setfill等等。 15. C++语言支持的两种多态性分别是编译时的多态性和___的多态性。 答案:运行时 [解析]多态性包括静态的(编译时)多态性和动态的(运行时)多态性。 16. 设函数sum是由函数模板实现的,并且sum(3,6)和sum(4.6,8)都是正确的函数调用,则函 数模板具有___个类型参数。 答案:2 17. 执行下列代码 string str(\ cout< [解析]substr取子字符串,第1个参数表示要截取子串在字符串中的位置,第2个表示取多少个 字符。 18. 在面向对象的程序设计中,将一组对象的共同特性抽象出来形成___。 答案:类 [解析]类是相似特征的对象的抽象,对象是类的一个实例。 19. 定义类动态对象数组时,元素只能靠自动调用该类的___来进行初始化。 答案:无参构造函数 [解析]使用new 创建动态对象数组,不能有参数,所以只能调用无参的构造函数,初始化对象 20. 已知有20个元素int类型向量V1,若用V1初始化为V2向量,语句是___。 答案:ector [解析]采用向量初始化另一个向量的形式:vector 三、改错题(本大题共5小题,每小题2分,共10分)下面的类定义中有一处错误,请用下 横线标出错误所在行并给出修改意见。 1. #include Test(int i,int j){x=i,y=j;} int getx(){return x;} int gety(){return y;} }; void main() {Test mt(10,20); cout< } 答案:int x,y=20;在类内部不能对数据成员直接赋值。 [修改]int x,y; 2. #include fun(int i,int j) {x=i;y=j;} show() {cout<<\if(y) cout<<\cout< void main() {Test a; a.fun(1); a.show(); a.fun(2,4); a.show(); } 答案:int i,int j调用时,既有一个参数,也有两个参数,且没有重载,所以参数需要带默认 值。所以int i,int j错误。 [修改]int i,int j=0//注j只要有一个int类型的数据就行。 3. #include virtual void fun()=0; A(int a) {i=a;} }; class B:public A {int j; public: void fun() {cout<<\\n\B(int m,int n=0):A(m),j(n){} }; void main() {A *pa; B b(7); pa=&b; } 答案:B(int m,int n=0):A(m),j(n){}因为基类是抽象类,不能被实例化,所以在派生类中不能 调用初始化基类对象。所以B(int m,int n=0):A(m),j(n){}错误,删去A(m)。 [修改]B(int m,int n=0):j(n){} 4. #include {cout< cout< void fun(X); }; void fun(X t) {cout< 答案:cout< Bas(char *s=\\0\void show(); protected: char name[20]; }; Bas b; void show() {cout<<\void main() {Bas d2(\show(); } 答案:void show();是普通函数不是成员函数,但是要访问类成员,需要定义为友元函数。 [修改]friend void show(); 四、完成程序题(本大题共5小题,每小题4分,共20分) 1. 在下面程序横线处填上适当字句,以使该程序执行结果为: 50 4 34 21 10 0 7.1 8.1 9.1 10.1 11.1 #include for (int i=0;i t=a[i], a[i]=a[n-1-i], a[n-1-i]=t; } void main () {int a[5]={10,21,34,4,50}; double d[6]={11.1,10.1,9.1,8.1,7.1}; f(a,5);f(d,6); for (int i=0;i<5;i++) cout < cout << d[i]<< \cout << endl; } 答案:T a[],int n,T t=0; [解析]不同的数据类型的调用,使用了模板。f函数增加t变量,因为实参类型不同,所以t的 类型应该是T类型的。 2. 在下面程序的底画线处填上适当的字句,使该程序执行结果为40。 #include int Getnum() {return Test::x+7;} }; _______; void main() {Test test; cout< 答案:static int x;,int Test::x=30; [解析]从成员函数访问方式类名::成员可知是静态成员所以static int x;从结果要对初始 化为30,且在类外进行初始化, int Test::x=30;。 3. 在下列程序的空格处填上适当的字句,使输出为:0,2,10。 #include class Magic {double x; public: Magic(double d=0.00):x(fabs(d)) {} Magic operator+(______) { return Magic(sqrt(x*x+c.x*c.x)); } _______operator<<(ostream & stream,Magic & c) { stream< void main() {Magic ma; cout< 答案:operator+(Magic&c),friend ostream&operator [解析]对加法进行重载,operator+(Magic & c),是对插入符进行重载,要访问成员所以定义 为友元函数,friend ostream & operator。 4. 下面是一个输入半径,输出其面积和周长的C++程序,在下划线处填上正确的语句。 #include double l=2.0*pi*rad; double s=pi*rad*rad; cout<<\\n The long is:\cout<<\:\ 答案:using namespace std,#define pi 3.14159 [解析]进行输入或输出要引入iostream, 所以using namespace std;从标点看没有分号,所以 使用宏定义,#define pi 3.14159。 5. 程序实现大写字母转换成小写字母。 #include cin>>a; if(_______) a=a+i; cout< 答案:int i=32;,a>=A && a<=Z [解析]大写字母变小写字母相差32,需要对i声明并初始化。大写字母变小写字母。要判断字 符是大写字母。 五、程序分析题(本大题共4小题,每小题5分,共20分) 1. 给出下面程序输出结果。 #include virtual void print() {cout<< \}; class b:public a {}; class c:public b {public: void print(){cout<<\}; void show(a *p) {(*p).print(); } void main() {a a; b b; c c; show(&a); show(&b); show(&c); } 答案:a prog... a prog... c prog... [解析]考查多态性的。a类对象调用本身的虚函数,b类因为没有覆写print,所以仍然调用基 类的虚函数。而c类重新定义print虚函数,所以调用c类的print。 2. 给出下面程序输出结果。 #include bool fun(long n); void main() {long a=10,b=30,l=0; if(a%2==0) a++; for(long m=a;m<=b;m+=2) if(fun(m)) {if(l++==0) cout < cout < bool fun(long n) {int sqrtm=(int)sqrt(n); for(int i=2;i<=sqrtm;i++) if(n%i==0) return false; return true; } 答案:11 13 17 19 23 29 [解析]循环体用来判断n是否是质数的函数,在main函数判断10~30之间质数。 3. 给出下面程序输出结果。 #include Test(int i,int j=0) {x=i;y=j;} int get(int i,int j) {return i+j;} }; void main() {Test t1(2),t2(4,6); int (Test::*p)(int,int=10); p=Test::get; cout<<(t1.*p)(5)< cout<<(p1->*p)(7,20)< 答案:15 27 [解析]指向类成员函数的指针的使用,*p指向Test类中有两个参数的函数的一个指针。 P=Test::get.这样p就和get发生了联系。(t1.*p)(5)等价于调用一个参数的get函数。 4. #include class student {char name[8]; int deg; char level[7]; friend class process; // 说明友元类 public: student(char na[],int d) { strcpy(name,na); deg=d; } }; class process { public: void trans(student &s) {int i=s.deg/10; switch(i) {case 9: strcpy(s.level, \优\case 8: strcpy(s.level,\良\case 7: strcpy(s.level,\中\case 6: strcpy(s.level,\及格\default: strcpy(s.level,\不及格\} } void show(student &s) {cout< void main() { student st[]={student(\张三\李四\王五 \孙六\process p; cout<<\结果:\姓名\成绩\等级\for(int i=0;i<4;i++) { p.trans(st[i]); p.show(st[i]);} } 答案:结果:姓名成绩等级 张三78中 李四92优 王五62及格 孙六88良 六、程序设计题(本大题共1小题,共10分) 1. 已定义一个Shape抽象类,在此基础上派生出矩形Rectangle和圆形Circle类,二者都有 GetPerim()函数计算对象的周长,并编写测试main()函数。 class Shape {public: Shape(){} ~Shape(){} virtual float GetPerim()=0; } 答案:class Rectangle:public Shape {public: Rectangle(float i,float j):L(i),W(j){} ~Rectangle(){} float GetPerim(){return 2*(L+W);} private: float L,W; }; class Circle:public Shape {public: Circle(float r):R(r){} float GetPerim(){return 3.14*2*R;} private: float R; }; void main() {Shape * sp; sp=new Circle(10); cout< cout< C++程序设计模拟试卷(二) 一、单项选择题(本大题共20小题,每小题1分,共20分)在每小题列出的四个备选项中 只有一个是符合题目要求的,请将其代码填写在题后的括号内。错选、多选或未选均无 分。 1. 静态成员函数没有() A. 返回值 B. this指针 C. 指针参数 D. 返回类型 答案:B 解析:静态成员函数是普通的函数前加入static,它具有函数的所有的特征:返回类型、 形参,所以使用(P107)静态成员函数,指针可以作为形参,也具有返回值。静态成员是类具有的 属性,不是对象的特征,而this表示的是隐藏的对象的指针,因此静态成员函数没有this 指针 。静态成员函数当在类外定义时,要注意不能使用static关键字作为前缀。由于静态成员函数在 类中只有一个拷贝(副本),因此它访问对象的成员时要受到一些限制:静态成员函数可以直接 访问类中说明的静态成员,但不能直接访问类中说明的非静态成员;若要访问非静态成员时,必 须通过参数传递的方式得到相应的对象,再通过对象来访问。 2. 假定AB为一个类,则执行“AB a(2), b[3],*p[4];”语句时调用该类构造函数的次数 为() A. 3 B. 4 C. 5 D. 9 答案:B 解析:a(2)调用1次带参数的构造函数,b[3]调用3次无参数的构造函数,指针没有给它 分配空间,没有调用构造函数。所以共调用构造函数的次数为4。 3. 有关多态性说法不正确的是() A. C++语言的多态性分为编译时的多态性和运行时的多态性 B. 编译时的多态性可通过函数重载实现 C. 运行时的多态性可通过模板和虚函数实现 D. 实现运行时多态性的机制称为动态多态性 答案:C 解析:多态性分为静态的和动态的。静态通过函数的重载来实现,动态是通过基类指针或 基类引用和虚函数来实现的。所以错误的是C项。 4. 假定一个类的构造函数为“A(int i=4, int j=0) {a=i;b=j;}”, 则执行“A x (1);”语 句后,x.a和x.b的值分别为() A. 1和0 B. 1和4 C. 4和0 D. 4和1 答案:A 解析:带默认的构造函数,对应实参没有值时就采用形参值。调用构造函数时,i=1,不采 用默认值,而只有一个参数,j采用默认值0即j=0,因此a=1,b=0,选择A项。 5. 类MyA的拷贝初始化构造函数是() A. MyA() B. MyA(MyA*) C. MyA(MyA&) D. MyA(MyA) 答案:C 解析:复制即拷贝构造函数使用对象的引用作形参,防止临时产生一个对象,A无参构造函 数,B是指针作为形参,D项是对象,所以选择C项。 6. 在C++中,函数原型不能标识() A. 函数的返回类型 B. 函数参数的个数 C. 函数参数类型 D. 函数的功能 答案:D 解析:函数的声明,说明函数的参数、返回类型以及函数名,函数体即实现部分决定功能。所以 函数的原型不能决定函数的功能。 7. 友元关系不能() A. 提高程序的运行效率 B. 是类与类的关系 C. 是一个类的成员函数与另一个类的关系 D. 继承 答案:D 解析:友元可以是函数与类的关系即友元函数,也可以类与类的关系即友元类,但友元不 能继承,是单向性,且不具有传递性。友元可以访问类中所有成员,提高了访问的方便性。因此 选择D项。 8. 实现两个相同类型数加法的函数模板的声明是() A. add(T x,T y) B. T add(x,y) C. T add(T x,y) D. T add(T x,T y) 答案:D 解析:实现两个相同类型数加法结果应该和操作数具有相同类型。进行加法运算后结果也 是和参数具有相同类型,需要返回值。A无返回值时要用void,B形参无类型,C形参y没有类型 ,所以选择D项。 9. 在int a=3,int *p=&a;中,*p的值是() A. 变量a的地址值 B. 无意义 C. 变量p的地址值 D. 3 答案:D 解析:*p代表引用a变量的值,p代表a的地址值。所以选择D项。 10. 下列不是描述类的成员函数的是() A. 构造函数 B. 析构函数 C. 友元函数 D. 拷贝构造函数 答案:C 解析:构造函数、析构函数、拷贝构造函数都是特殊的成员函数,友元则不是成员函数。 所以选择C项。 11. 如果从原有类定义新类可以实现的是() A. 信息隐藏 B. 数据封装 C. 继承机制 D. 数据抽象 答案:C 解析:继承指在原有类的基础上产生新类。数据封装即数据和操作组合在一起,形成类。 信息的隐藏,通过访问权限来实现。数据抽象,将事物的特征抽象为数据成员或服务。因此选择 C项。 12. 下面有关类说法不正确的是() A. 一个类可以有多个构造函数 B. 一个类只有一个析构函数 C. 析构函数需要指定参数 D. 在一个类中可以说明具有类类型的数据成员 答案:C 解析:构造函数可以有参数、可以重载、因此可以有多个,A项正确。析构函数只有一个不 能重载、不能继承,没有返回值,B项正确,C项错误。 13. 在函数定义中的形参属于() A. 全局变量 B. 局部变量 C. 静态变量 D. 寄存器变量 答案:B 解析:形参或函数中定义的变量都是局部变量。在函数外定义的变量是全局变量。形参只能用局 部变量,频繁使用的变量可以声明为寄存器变量,形参不能使用静态变量或寄存器变量。 14. 下列有关重载函数的说法中正确的是() A. 重载函数必须具有不同的返回值类型 B. 重载函数参数个数必须相同 C. 重载函数必须有不同的形参列表 D. 重载函数名可以不同 答案:C 解析:函数的重载必须函数名相同而形参类型或个数不同,与返回值无关。 15. this指针存在的目的是() A. 保证基类私有成员在子类中可以被访问 B. 保证基类保护成员在子类中可以被访问 C. 保证每个对象拥有自己的数据成员,但共享处理这些数据成员的代码 D. 保证基类公有成员在子类中可以被访问 答案:C 解析:C++要求函数在被调用之前,应当让编译器知道该函数的原型,以便编译器利用函数 原型提供的信息去检查调用的合法性,强制参数转换成为适当类型,保证参数的正确传递。对于 标准库函数,其声明在头文件中,可以用#include宏命令包含这些原型文件;对于用户自定义函 数,先定义、后调用的函数可以不用声明,但后定义、先调用的函数必须声明。一般为增加程序 的可理解性,常将主函数放在程序开头,这样需要在主函数前对其所调用的函数一一进行声明 ,以消除函数所在位置的影响。所以选择C项。 16. 关于new运算符的下列描述中,错误的是() A. 它可以用来动态创建对象和对象数组 B. 使用它创建的对象或对象数组可以使用运算符delete删除 C. 使用它创建对象时要调用构造函数 D. 使用它创建对象数组时必须指定初始值 答案:D 解析:new创建的对象数组不能指定初始值,所以调用无参的构造函数,选择D项。 17. 已知:p是一个指向类A数据成员m的指针,A1是类A的一个对象。如果要给m赋值为5,正确 的是() A. A1.p=5; B. A1->p=5; C. A1.*p=5; D. *A1.p=5; 答案:C 解析:A中p是指针即地址,错误;B选项中A1不是指针不能使用指向运算符->,错误 ;“*”比“.”级别要高,所以D选项*A1.p=5相当于(*A1).p=5;错误。另外涉及到指向成员函数 时注意以下几点: 指向成员函数的指针必须于其赋值的函数类型匹配的三个方面:(1)参数类型和个数;(2)返回 类型;(3)它所属的类类型。 成员函数指针的声明:指向short型的Screen类的成员的指针定义如下: short Screen::* ps_Screen; ps_Screen可以用_height的地址初始化如下:short Screen::*ps_Screen=&Screen::_height; 类成员的指针必须总是通过特定的对象或指向改类型的对象的指针来访问。是通过使用两个指 向成员操作符的指针(针对类对象和引用的.*,以及针对指向类对象的指针的->*)。 18. 以下基类中的成员函数表示纯虚函数的是() A. virtual void tt()=0 B. void tt(int)=0 C. virtual void tt(int) D. virtual void tt(int){} 答案:A 解析:当在基类中不能为虚函数给出一个有意义的实现时,可以将其声明为纯虚函数,实 现由派生类完成。格式:virtual<函数返回类型说明符><函数名>(<参数表>)=0;。 19. C++类体系中,不能被派生类继承的有() A. 常成员函数 B. 构造函数 C. 虚函数 D. 静态成员函数 答案:B 解析:构造函数不能被继承。 20. 静态成员函数不能说明为() A. 整型函数 B. 浮点函数 C. 虚函数 D. 字符型函数 答案:C 解析:使用关键字static声明的成员函数就是静态成员函数,静态成员函数也属于整个类 而不属于类中的某个对象,它是该类的所有对象共享的成员函数。 静态成员函数可以在类体内定义,也可以在类外定义。当在类外定义时,要注意不能使用 static关键字作为前缀。 由于静态成员函数在类中只有一个拷贝(副本),因此它访问对象的成员时要受到一些限制:静 态成员函数可以直接访问类中说明的静态成员,但不能直接访问类中说明的非静态成员;若要访 问非静态成员时,必须通过参数传递的方式得到相应的对象,再通过对象来访问。虚函数是非静 态的、非内联的成员函数。静态成员函数不能被说明为虚函数。 二、填空题(本大题共20小题,每小题1分,共20分)请在每小题的空格中填上正确答案 。错填、不填均无分。 1. 假设int a=1,b=2;则表达式(++a/b)*b--的值为___。 答案:2 [解析]前缀++或――表示先使变量值变化,再使用,这和后缀恰恰相反。但是编译 cout<<(++a/b)*b--时,先++a/b值为1,后1*b--,先取b=2,结果为2,再让b=1。 2. 抽象类中至少要有一个___函数。 答案:纯虚 [解析]至少有一个纯虚函数的类就称为抽象类,即不能实例化。 3. 一个抽象类的派生类可以实例化的必要条件是实现了所有的___。 答案:纯虚函数的定义 [解析]抽象类只因有纯虚函数,所以不能被实例化,所以派生类要实例化必须对纯虚函数进行 定义。 4. 下面程序的输出结果为___。 #include cout< 答案:5 [解析]do-while循环,前缀先使i减少1后判断是否为零,不为零时再次执行循环,为零退出 循环。循环值执行3次就退出,所以结果为5。 5. 静态成员函数、友元函数、构造函数和析构函数中,不属于成员函数的是___。 答案:友元函数 [解析]友元函数不是类成员,但可以访问类成员。类的封装性保证了数据的安全,但引入友元 ,虽然访问类是方便了,但确实破坏类访问的安全性。 6. 在用C++进行程序设计时,最好用___代替malloc。 答案:new [解析]new与delete是C++语言特有的运算符,用于动态分配和释放内存。new用于为各种数据 类型分配内存,并把分配到的内存首地址赋给相应的指针。new的功能类似于malloc()函数。 使用new的格式为: <指针变量>new<数据类型>; 其中,<数据类型>可以是基本数据类型,也可以是由基本类型派生出来的类型;<指针变量>取得 分配到的内存首地址。new有3种使用形式。 (1)给单个对象申请分配内存 int *ip;ip=new int;//ip指向1个未初始化的int型对象 (2)给单个对象申请分配内存的同时初始化该对象 int *ip;ip=new int(68);//ip指向1个表示为68的int型对象 (3)同时给多个对象申请分配内存 int *ip;ip=new int[5];//ip指向5个未初始化的int型对象的首地址 for(int i=0;i<5;i++)ip[i]=5*i+1;//给ip指向的5个对象赋值 用new申请分配内存时,不一定能申请成功。若申请失败,则返回NULL,即空指针。因此,在程 序中可以通过判断new的返回值是否为0来获知系统中是否有足够的空间供用户使用。 7. 由const修饰的对象称为___。 答案:常对象 [解析]使用const关键字说明的成员函数称为常成员函数,使用const关键字说明的对象称为常 对象。 常成员函数的说明格式如下:<返回类型说明符><成员函数名>(<参数表>)const; 常成员函数不更新对象的数据成员,也不能调用该类中没有用const修饰的成员函数。常对象 只能调用它的常成员函数,而不能调用其他成员函数。const关键字可以用于参与重载函数的区 分。 8. 在C++程序设计中,建立继承关系倒挂的树应使用___继承。 答案:单 [解析]一个基类可以派生多个子类,一个子类可以再派生出多个子类,这样就形成了一个倒立 的树。 9. 基类的公有成员在派生类中的访问权限由___决定。 答案:访问控制方式或继承方式 10. 不同对象可以调用相同名称的函数,但执行完全不同行为的现象称为___。 答案:多态性 [解析]多态性的概念。虚函数是实现多态的基础,运行过程中的多态需要同时满足3个条件 :(1)类之间应满足子类型关系。(2)必须要有声明的虚函数。(3)调用虚函数操作的是指向对象 的指针或者对象引用;或者是由成员函数调用虚函数(如果是在构造函数或析构函数中调用虚函 数,则采用静态联编)。 11. this指针始终指向调用成员函数的___。 答案:对象 this指针是隐藏的指针,它指向调用函数的对象。 12. 预处理命令以___符号开头。 答案:operater [解析]文件包含、预处理和编译都是以#开头。 13. 类模板用来表达具有___的模板类对象集。 答案:相同处理方法 [解析]模板特点是不同的数据具有相同的处理方法的抽象。 14. C++程序的源文件扩展名为___。 答案:cpp [解析]源程序*.cpp,目标文件为*.obj,可执行程序*.exe。 15. 在#include命令中所包含的头文件,可以是系统定义的头文件,也可以是___的头文件。 答案:自定义 [解析]#include装入文件有两种方式<>和“”,一是系统的,一是自定义文件。 16. vector类中向向量尾部插入一个对象的方法是___。 答案:push_back 17. C++语言中如果调用函数时,需要改变实参或者返回多个值,应该采取___方式。 答案:传地址或引用 [解析]传地址即指针,在函数中通过指针修改它指向的变量的值时,实参也就变化了。使用引 用,直接修改变量的别名即引用的值,该变量也就随着变化。 18. 语句序列 ifstream infile; infile.open(“data.dat”); 的功能可用一个语句实现,这个语句是___。 答案:ifstream infile(“data.dat”); [解析]void ifstream::open(const char *fname,int mode=ios::in,int access=filebuf::openprot); ifstream::ifstream(const char *fname,int mode=ios::in,int access=filebuf::openprot); 其中,第一个参数是用来传递文件名的;第二个参数mode的值决定文件将如何被打开;第三个参 数access的值决定文件的访问方式,一般取缺省值filebuf::openprot,表示是普通文件。 mode的取值如下:(1)ios::in:打开一个文件进行读操作,而且该文件必须已经存在 ;(2)ios::nocreate:不建立新的文件。当文件不存在时,导致open()失败 ;(3)ios::noreplace:不修改原来已经存在的文件。若文件已经存在,导致open()失败 ;(4)ios::binary:文件以二进制方式打开,缺省时为文本文件。 19. 如果要把类B的成员函数void fun()说明为类A的友元函数,则应在类A中加入语句___。 答案:(P111)friend void B::fun(); [解析]声明成员函数作为另外一个类的友元函数时,使用类作用域运算符::。 20. 在编译指令中,宏定义使用___指令。 答案:#define [解析]静态成员是所有对象共享的特征,也就是类的特征。 三、改错题(本大题共5小题,每小题2分,共10分)下面的类定义中有一处错误,请用下 横线标出错误所在行并给出修改意见。 1. #include using namespace std; class A {public: A(const char *na){strcpy(name,na);} private: char name[80]; }; class B:public A { public: B(const char *nm):A(nm){} void show(); }; void B::show() { cout<<\} void main() { B b1(\b1.show(); } 答案:private:因为name如果是私有的,在派生类中无法访问,而基类没有提供成员函数来访问 name,所以更改name访问权限为公有或保护,这样对于派生类来说是透明的。 [修改]public:或protected: 2. #include for(;i a[i]=a[n-i-1]; a[n-i-1]=k; } } void show(int a[],int n) {for(int i=0;i void main() {int p[5]; int i=0,n=5; for(;i<5;i++) {p[i]=i;} f(*p,n); show(p,n); 答案:[修改]f(p,n); [解析]f(*p,n);f函数第一个参数是指针而调用时使用*p,*p表示p所指向的变量或对象,不是 地址即不是指针。 3. #include cout<<*p< 答案:int * const p=&i;在指针变量前加const表示一个常指针即地址不能变化,它指向的变 量不能改变且定义时必须设置指向变量或对象的地址。 [修改]int *p=&i; 4. #include cout<<*p< 答案:*p=i;指针即地址没有被赋值。 [修改]p=&i; 5. #include void fun(int i,int j) {x=i;y=j;} void show() {cout< void main() {A a1; a1.fun(2); a1.show(); } 答案:void fun(int i,int j)调用时有一个参数,形参有两个,可以使第二个带默认值。 [修改]void fun(int i,int j=0) 四、完成程序题(本大题共5小题,每小题4分,共20分) 1. 完成下面类中成员函数的定义。 #include str & operator=(____) {delete st; set(a.st); return *this; } void show(){cout< void set(char *s)//初始化st {_____ strcpy(st,s); } }; void main() {str s1(\s1.show(),s2.show(); s2=s1; s1.show(),s2.show();} 答案:str &a,st=new char[strlen(s)+1]; [解析]对“=”运算符进行重载,调用时s2=s1,都是对象,所以形参使用对象的引用,不 要使用对象作为形参(产生临时对象)。使用strcpy进行字符的复制,st必须有一定的空间,空 间是strlen(s)+1(‘\0’作为结束符,strlen得到的长度不包括结束符)。 2. 一个类的头文件如下所示,num初始化值为5,程序产生对象T,且修改num为10,并使用 show()函数输出num的值10。 #include static int num; public: Test(int); void show(); }; _________ Test::Test(int n) {num=n;} void Test::show() {cout< 答案:int Test::num=5;,t.show(); [解析]静态成员在类外初始化,注意它的格式。调用show输出。 3. 下面是一个三角形三边,输出其面积C++程序,在下划线处填上正确的语句。 #include cout<<\________ if(a+b>c&&a+c>b&&c+b>a) {double l=(a+b+c)/2; _______ cout<<\} else cout<<\} void main() {area();} 答案:cin>>a>>b>>c;,double s=sqrt(l*(l-a)*(l-b)*(l-c)); [解析]输入三个边的长度,由公式得出三角形的面积double s=sqrt(l*(l-a)*(l-b)*(l-c)); 4. 下面程序中Base是抽象类。请在下面程序的横线处填上适当内容,以使程序完整,并使程序 的输出为: Der1 called! Der2 called! #include _______________; }; class Der1:public Base {public: void display(){cout<<\}; class Der2:public Base {public: void display(){cout<<\}; void fun(______________) {p->display();} void main() {Der1 b1; Der2 b2; Base * p=&b1; fun(p); p=&b2; fun(p); } 答案:virtual void display()=0;,Base *p [解析]抽象类有纯虚函数,派生类为display。结果fun函数用指针做参数。 5. 下面程序中用来求数组和。请在下面程序的横线处填上适当内容,以使程序完整,并使程序 的输出为:s=150。 #include Arr():a(0),n(0){} Arr(int *aa, int nn) {n=nn; a=new int[n]; for(int i=0;i ~Arr(){delete a;} _____________; {return *(a+i);} }; void main() {int b[5]={10,20,30,40,50}; Arr a1(b,5); int i=0,s=0; _____________ s+=a1.GetValue(i); cout<<\} 答案:int GetValue(int i),for(;i<5;i++) [解析]函数调用GetValue,由此可知要定义该函数,循环求和,循环5次。 五、程序分析题(本大题共4小题,每小题5分,共20分) 1. 给出下面程序输出结果。 #include example(int b=5){a=b++;} void print(){a=a+1;cout < void main() {example x; const example y(2); x.print(); y.print(); } 答案:62 [解析]x是普通对象,调用普通的print函数;而y常对象,调用常成员函数。 2. 给出下面程序输出结果。 #include int **p2=&p1; int b=20; p1=&b; cout<<**p2< 答案:20 [解析]p1指向b,而p指向p1的地址。*p2表示p1的地址,p1的地址就是&b,即*p2是&b,所以 **p2就是b变量的值。 3. 给出下面程序输出结果。 #include Base(int y=0) {Y=y;cout<<\\n\~Base() {cout<<\\n\void print() {cout < class Derived:public Base {private: int Z; public: Derived (int y, int z):Base(y) {Z=z; cout<<\\n\} ~Derived() {cout<<\~Derived()\n\void print() {Base::print(); cout< void main() {Derived d(10,20); d.print(); } 答案:Base(10) Derived(10,20) 10 20 ~Derived() ~Base() [解析]派生类对象,先调用基类构造函数输出Base(10),后调用派生类构造函数输出 Derived(10,20),后执行d.print(),调用派生类的print,再调用Base::print()输出10,后返回 输出z的值20。后派生类析构,再基类析构。 4. 给出下面程序输出结果。 #include {cout<<\构造函数\n\virtual void fun() {cout<<\函数\n\}; class B:public A {public: B() {cout<<\构造函数\n\ void fun() {cout<<\函数\n\}; void main() {B d;} 答案:A构造函数 A::fun()函数 B构造函数 B::fun()calle函数 [解析]定义派生类对象,首先调用基类构造函数,调用A类中fun(),然后调用B类的构造函数 ,在调用B的fun函数。 六、程序设计题(本大题共1小题,共10分) 1. 编写类String的构造函数、析构函数和赋值函数和测试程序。 已知类String的原型为: #include String(const char *str=NULL); // 普通构造函数 String(const String &other); // 拷贝构造函数 ~String(); // 析构函数 String & operator=(const String &other); // 赋值函数 void show() {cout< private: char *m_data; // 用于保存字符串 }; 答案:String::~String() {delete[]m_data;//由于m_data是内部数据类型,也可以写成delete m_data; } String::String(const char *str) {if(str==NULL) {m_data=new char[1];//若能加NULL判断则更好 *m_data=\0; } else {int length=strlen(str); m_data=new char[length+1]; //若能加NULL判断则更好 strcpy(m_data, str); } } String::String(const String &other) {int length=strlen(other.m_data); m_data=new char[length+1];//若能加NULL判断则更好 strcpy(m_data, other.m_data); } String & String::operator=(const String &other) {if(this==&other) return *this; delete[]m_data; int length=strlen(other.m_data); m_data=new char[length+1];//若能加NULL判断则更好 strcpy(m_data, other.m_data); return *this; } void main() {String str1(\str1.show(); str2=str1; str2.show(); String str3(str2); str3.show(); }__ C++程序设计模拟试卷(三) 一、单项选择题(本大题共20小题,每小题1分,共20分)在每小题列出的四个备选项中 只有一个是符合题目要求的,请将其代码填写在题后的括号内。错选、多选或未选均无 分。 1. 设有定义int i;double j=5;,则10+i+j值的数据类型是() A. int B. double C. float D. 不确定 答案:B 解析:考察数据的转换,j是double类型,运算只能作同类型的运算,所以要转换,而int能自动 转换为double类型,所以结果是double类型。 2. 要禁止修改指针p本身,又要禁止修改p所指向的数据,这样的指针应定义为() A. const char *p=“ABCD”; B. char *const p=“ABCD”; C. char const *p=“ABCD”; D. const char * const p=“ABCD”; 答案:D 解析:const char *p说明禁止通过p修改所指向的数据。char * const p则说明不能修改 指针p的地址。因此const char * const p=“ABCD”;它禁止修改指针p本身,又禁止修改p 所指 向的数据。 3. 类的构造函数被自动调用执行的情况是在定义该类的() A. 成员函数时 B. 数据成员时 C. 对象时 D. 友元函数时 答案:C 解析:建立对象时,自动构造函数的初始化对象,是系统自动调用的。而成员函数、友元 函数,需要用户直接调用,因此选择C项。 4. 已知类A是类B的友元,类B是类C的友元,则() A. 类A一定是类C的友元 B. 类C一定是类A的友元 C. 类C的成员函数可以访问类B的对象的任何成员 D. 类A的成员函数可以访问类B的对象的任何成员 答案:C 解析:友元说明方法如下: friend?<类名>;//友元类类名 使用友元可以访问所有成员: (1)友元关系不能被继承。 (2)友元关系是单向的,不具有交换性。所以,B项和D项错误。 (3)友元关系不具有传递性。所以,A项错误。 5. 假定一个类的构造函数为“A(int i=4, int j=0) {a=i;b=j;}”, 则执行“A x (1);”语 句后,x.a和x.b的值分别为() A. 1和0 B. 1和4 C. 4和0 D. 4和1 答案:A 解析:带默认的构造函数,对应实参没有值时就采用形参值。调用构造函数时,i=1,不采 用默认值,而只有一个参数,j采用默认值0即j=0,因此a=1,b=0,选择A项。 6. 关于this指针使用说法正确的是() A. 保证每个对象拥有自己的数据成员,但共享处理这些数据的代码 B. 保证基类私有成员在子类中可以被访问。 C. 保证基类保护成员在子类中可以被访问。 D. 保证基类公有成员在子类中可以被访问。 答案:A 解析:this指针是隐藏的,可以使用该指针来访问调用对象中的数据。基类的成员在派生 类中能否访问,与继承方式有关,与this没有关系。所以选择A项。 7. 所谓多态性是指() A. 不同的对象调用不同名称的函数 B. 不同的对象调用相同名称的函数 C. 一个对象调用不同名称的函数 D. 一个对象调用不同名称的对象 答案:B 解析:多态性有两种静态多态性和动态多态性,静态多态性是指调用同名函数,由于参数 的不同调用不同的同名函数;动态多态性是指不同对象调用同名函数时,由于对象不同调用不同 的同名函数。多态性肯定具有相同的函数名,所以选择B项。 8. 友元关系不能() A. 提高程序的运行效率 B. 是类与类的关系 C. 是一个类的成员函数与另一个类的关系 D. 继承 答案:D 解析:友元可以是函数与类的关系即友元函数,也可以类与类的关系即友元类,但友元不 能继承,是单向性,且不具有传递性。友元可以访问类中所有成员,提高了访问的方便性。因此 选择D项。 9. 语句ofstream f(“TEMP.DAT”,ios::app|ios::binary)?的功能是建立流对象f,试图打 开文件TEMP.DAT 并与之连接,并且() A. 若文件存在,将文件写指针定位于文件尾;若文件不存在,建立一个新文件 B. 若文件存在,将其置为空文件;若文件不存在,打开失败 C. 若文件存在,将文件写指针定位于文件首;若文件不存在,建立一个新文件 D. 若文件存在,打开失败;若文件不存在,建立一个新文件 答案:A 解析:ios::binary,采用二进制形式,ios::app定位到文件尾部。 10. 构造函数不具备的特征是() A. 构造函数的函数名与类名相同 B. 构造函数可以重载 C. 构造函数可以设置默认参数 D. 构造函数必须指定类型说明 答案:D 解析:构造函数无返回类型不能继承但可以重载,所以选择D项。 11. 在公有继承的情况下,基类的公有或保护成员在派生类中的访问权限() A. 受限制 B. 保持不变 C. 受保护 D. 不受保护 答案:B 解析:继承方式的不同派生类成员的权限也不同,采用公有继承,除了私有无法访问外 ,公有、保护在派生类中保持不变,所以选择B项。 12. 假定一个类的构造函数为A(int aa,int bb) {a=aa--;b=a*bb;},则执行A x(4,5);语句后 ,x.a和x.b的值分别为() A. 3和15 B. 5和4 C. 4和20 D. 20和5 答案:C 解析:a=4,因为后减,b的值与a、bb相关,b=4*5=20,而与aa没有任何关系。 13. C++对C语言做了很多改进,即从面向过程变成为面向对象的主要原因是() A. 增加了一些新的运算符 B. 允许函数重载,并允许设置缺省参数 C. 规定函数说明符必须用原型 D. 引进了类和对象的概念 答案:D 解析:C++是一面向对象的语言,面向对象的特征,抽象、多态、继承和封装。 14. 在类中说明的成员可以使用关键字的是() A. public B. extern C. cpu D. register 答案:A 解析:extern用于声明外部变量的。register声明寄存器类型变量。无cpu类型。它们都不能声 明类成员。public声明为公有访问权限,所以选择A项。 15. C++语言中所有在函数中定义的变量,连同形式参数,都属于() A. 全局变量 B. 局部变量 C. 静态变量 D. 函数 答案:B 解析:变量存储类可分为两类:全局变量和局部变量。 (1)全局变量:在函数外部定义的变量称为全局变量,其作用域为:从定义变量的位置开始到 源程序结束。使用全局变量降低了程序的可理解性,软件工程学提倡尽量避免使用全局变量。 (2)局部变量:在函数内部定义的变量称为局部变量,其作用域为:从定义变量的位置开始到 函数结束。局部变量包含自动变量(auto)静态变量(static)以及函数参数。形参不能是静态 的。所以选择B项。 16. 在私有继承的情况下,基类成员在派生类中的访问权限() A. 受限制 B. 保持不变 C. 受保护 D. 不受保护 答案:A 解析:私有继承下,基类中的公有或保护成员在派生类中也是私有的,所以选择A选项。 17. 使用地址作为实参传给形参,下列说法正确的是() A. 实参是形参的备份 B. 实参与形参无联系 C. 形参是实参的备份 D. 实参与形参是同一对象 答案:D 解析:地址作为实参,表示实参与形参代表同一个对象。如果实参是数值,形参也是普通 变量,此时形参是实参的备份。所以选择D项。 18. C++的继承性允许派生类继承基类的() A. 部分特性,并允许增加新的特性或重定义基类的特性 B. 部分特性,但不允许增加新的特性或重定义基类的特性 C. 所有特性,并允许增加新的特性或重定义基类的特性 D. 所有特性,但不允许增加新的特性或重定义基类的特性 答案:A 解析:派生类有两类成员:一是基类,二是自身类。派生类中的成员不能访问基类中的私 有成员,可以访问基类中的公有成员和保护成员。 19. 对于int *pa[5];的描述,正确的是() A. pa是一个指向数组的指针,所指向的数组是5个int型元素 B. pa是一个指向某个数组中第5个元素的指针,该元素是int型变量 C. pa[5]表示某个数组的第5个元素的值 D. pa是一个具有5个元素的指针数组,每个元素是一个int型指针 答案:D 解析:指针数组:数组元素都是相同类型的指针,相同类型的指针是说指针所指向的对象 类型是相同的。例如,语句int *pa[5];定义了一个指针数组。在指针数组的定义中有两个运 算符:*和[],运算符[]的优先级高于*,所以*pa[5]等价于*(pa[5]),pa[5]表示一 个数组,而*表示后面的对象为指针变量,合在一起*pa[5]表示一个指针数组。该数组包含5个 元素,每个元素都是指向int型的指针。所以选择D选项。 20. 以下基类中的成员函数表示纯虚函数的是() A. virtual void tt()=0 B. void tt(int)=0 C. virtual void tt(int) D. virtual void tt(int){} 答案:A 解析:当在基类中不能为虚函数给出一个有意义的实现时,可以将其声明为纯虚函数,实 现由派生类完成。格式:virtual<函数返回类型说明符><函数名>(<参数表>)=0;。 二、填空题(本大题共20小题,每小题1分,共20分)请在每小题的空格中填上正确答 案。错填、不填均无分。 1. 单目运算符作为类成员函数重载时,形参个数为___个。 答案:(P189)0 [解析]单目运算符使用成员函数重载可以不用形参,双目运算符使用一个参数。 2. 抽象类中至少要有一个___函数。 答案:(P173)纯虚 [解析]至少有一个纯虚函数的类就称为抽象类,即不能实例化。 3. 设类A有成员函数void f(void);若要定义一个指向类成员函数的指针变量pf来指向f,该 指针变量的声明语句是:___。 答案:(P117)void (A::*pf)(void)=&A::f; [解析]void(A::*pf)(void)=&A::f;指向成员函数的指针,它相当于两条语句 :void(A::*pf)(void);和pf=&A::f;。 4. 执行下列程序 double a=3.1415926,b=3.14; cout< [解析]题目设置精度即有效数字都是5,a四舍五入是3.1416,b是3.14。 5. vector类中用于删除向量中的所有对象的方法是___。 答案:clear() [解析]向量的使用。返回向量中对象的方法有:front()back()operator[],在向量中删 除对象的方法pop_back erase clear。 6. 重载的运算符保持其原有的___、优先级和结合性不变。 答案:操作数 [解析]运算符重载时要遵循以下规则: (1)除了类属关系运算符“.”、成员指针运算符“.*”、作用域运算符“::”、sizeof运算符 和三目运算符“?:”以外,C++中的所有运算符都可以重载。 (2)重载运算符限制在C++语言中已有的运算符范围内的允许重载的运算符之中,不能创建新的 运算符。 (3)重载之后的运算符不能改变运算符的优先级和结合性,也不能改变运算符操作数的个数及 语法结构。 7. 编译时的多态性通过___函数实现。 答案:重载 [解析]编译多态性,实现的方法主要通过函数的重载或运算符的重载。 8. 基类的公有成员在派生类中的访问权限由___决定。 答案:访问控制方式或继承方式 9. 假设类X的对象x是类Y的成员对象,则“Y Obj”语句执行时,先调用类___的构造函数。 答案:X [解析]派生类中的构造函数的执行顺序,先基类后派生类。 10. 下列程序段的输出结果是___。 cout.setf(ios::showpos); cout<<509.3< 答案:(P193)+509.3 [解析]输入、输出格式ios::showpos用于输出数据的符号位。 11. 下列程序段的输出结果是___。 for(i=0,j=10,k=0;i<=j;i++,j-=3,k=i+j);cout< [解析]for循环结构,三个表达式的作用,初始化、循环判断条件和循环变量变化。循环执行了 三次,k的作用是计算i、j的和。 12. C++中ostream的直接基类___。 答案:ios [解析]istream和ostream的直接基类是ios。 13. int n=0; while(n=1)n++; while循环执行次数是___。 答案:无限次 [解析]=是赋值运算符,不是关系运算符,且不等0,所以死循环。 14. C++中有两种继承:单继承和___。 答案:多继承 [解析]单继承和多继承,多继承即有多个基类。 15. 在C++中,利用向量类模板定义一个具有10个int的向量A,其元素均被置为1,实现此操作 的语句是___。 答案:vector [解析]定义向量列表vector 17. C++语言中如果调用函数时,需要改变实参或者返回多个值,应该采取___方式。 答案:传地址或引用 [解析]传地址即指针,在函数中通过指针修改它指向的变量的值时,实参也就变化了。使用引 用,直接修改变量的别名即引用的值,该变量也就随着变化。 18. 若函数的定义处于调用它的函数之前,则在程序开始可以省去该函数的___语句。 答案:声明 [解析]函数使用有两部分:声明和定义。定义在前,可以无声明;但函数定义在后,调用在前 的话,需要先声明函数的原型。 19. 在C++中有两种参数传递方式:传值和___。 答案:传引用 [解析](1)传值调用又分为数据传值调用和地址传值调用。(2)引用调用是将实参变量值传 递给形参,而形参是实参变量的引用名。引用是给一个已有变量起的别名,对引用的操作就是对 该引用变量的操作。 20. 将指向对象的引用作为函数的形参,形参是对象的引用,实参是 ___。 答案:对象名 [解析]实参与形参类型要一致,形参是对象的引用,实参应该是对象名。 三、改错题(本大题共5小题,每小题4分,共20分) 1. class ABC {int a; public: ABC(int aa)a(aa){} }; 答案:ABC(int aa)a(aa){}初始化列表格式错误。 [修改]ABC(int aa):a(aa){} 2. #include void init(int i,int j){zx=i;zy=j;} Aton(int i,int j,int n=0,int m=0) {X=i,Y=j,zx=m,zy=n; } }; void main() {Aton A(25,20,3,5); A.init(5,9); cout< 答案:int X,Y;因为X,Y都是私有的,在类外无法直接访问。 [修改]public:int X,Y; 3. #include ~Bas(){cout<<\virtual void f()=0; }; class Dev:public Bas {public: ~Dev(){cout<<\virtual void f(){cout<<\}; void main() {Bas *a=new Bas(); Dev p; a=&p; a->f(); } 答案:[修改]Bas *a; [解析]Bas *a=new Bas();抽象类不能被实例化,但可以声明指针或引用,所以不能用new,因 为new产生临时对象。 4. 以下程序实现交换a,b变量的值,请用下横线标出错误所在行并给出修改意见。 #include a=a-b; } void main() {int a=19,b=15; cout<<\swap(&a,&b); cout<<\} 答案:swap(&a,&b);函数的形参是变量的引用,调用时的实参应该是地址。 [修改]swap(a, b); 5. #include cout<<*p< 答案:int * const p=&i;在指针变量前加const表示一个常指针即地址不能变化,它指向的变 量不能改变且定义时必须设置指向变量或对象的地址。 [修改]int *p=&i; 四、完成程序题(本大题共5小题,每小题4分,共20分) 1. 在下面程序横线处填上适当内容,使程序执行结果为:\。 #include char * pdata; mystring(int len) {pdata=new char[len+1]; } ~mystring() {delete pdata;} void show(){cout< void fun(mystring** array,int len) {mystring*old=*array; _______; memcpy(*array, old, len); } void main() {mystring str(20); mystring*pstr=&str; mystring**ppstr=&pstr; strcpy(str.pdata,\fun(ppstr, 20); _______ } 答案:*array=new mystring(len);,(**ppstr).show();或str.show(); [解析]调用mystring类的构造函数开辟空间,后进行字符的复制。输出可以直接使用str或者 使用二级指针。 2. 在下面程序横线处填上适当字句,完成类的定义。 class line; class box { private: int color; int upx, upy; int lowx, lowy; public: _________ void set_color (int c){color=c;} void define_box (int x1, int y1, int x2, int y2) {upx=x1;upy=y1;lowx=x2;lowy=y2;} }; class line { private: int color; int startx, starty; int endx, endy; public: friend int same_color(line l,box b); void set_color (int c) {color=c;} void define_line (___________) {startx=x1;starty=y1;endx=x2;endy=y2;} }; int same_color(line l, box b) {if (l.color==b.color) return 1; return 0; } 答案:friend int same_color(line l, box b );,int x1, int y1, int x2, int y2 [解析]成员函数作为友元函数的使用。使用friend关键字。由函数体可知形参的类型和个数。 3. 下面程序用来求直角三角形斜边长度。 #include {private: double x,y; __________ public: Point(double i=0,double j=0) {x=i;y=j;} Point(Point &p) {x=p.x;y=p.y;} }; class Line {private: Point p1,p2; public: Line(Point &xp1,Point &xp2):________{} double GetLength(); }; double Line::GetLength() {double dx=p2.x-p1.x; double dy=p2.y-p1.y; return sqrt(dx*dx+dy*dy); } void main() { Point p1,p2(6,8); Line L1(p1,p2); cout< 答案:friend Line;,p1(xp1),p2(xp2) [解析]友元类的使用,定义Line是Point类的友元类,成员对象的初始化采用列表的形式。 4. 在下面程序的底画线处填上适当的字句,使该程序执行结果为40。 #include int Getnum() {return Test::x+7;} }; _______; void main() {Test test; cout< 答案:static int x;,int Test::x=30; [解析]从成员函数访问方式类名::成员可知是静态成员所以static int x;从结果要对初始 化为30,且在类外进行初始化, int Test::x=30;。 5. 在下列程序的空格处填上适当的字句,使输出为:0,2,10。 #include Magic(double d=0.00):x(fabs(d)) {} Magic operator+(______) { return Magic(sqrt(x*x+c.x*c.x)); } _______operator<<(ostream & stream,Magic & c) { stream< void main() {Magic ma; cout< 答案:operator+(Magic&c),friend ostream&operator [解析]对加法进行重载,operator+(Magic & c),是对插入符进行重载,要访问成员所以定义 为友元函数,friend ostream & operator。 五、程序分析题(本大题共2小题,每小题5分,共10分) 1. 运行程序,写出程序执行的结果。 #include cin>>a>>ch>>b>>c;//从键盘上输入1.5×c×10×20,×表示一个空格 cout< 答案:1 . 5 0 [解析]使用cin进行输入字符的输入的问题。1-->a,.-->ch,5-->b,空格转换为零给了c。 2. 给出下面程序输出结果。 #include class A {public: A() {cout<<\virtual ~A() {cout<<\virtual void f() {cout<<\void g() {f();} }; class B:public A {public: B() {f();cout<<\~B() {cout<<\}; class C:public B {public: C() {cout<<\~C() {cout<<\void f() {cout<<\}; void main() {A *a=new C; a->g(); delete a; } 答案:As f(). Bs cons. Cs cons. Cs f(). Cs des. Bs des. As des. [解析]定义C类对象时要调用基类构造函数从A到B再到C,调用B的构造函数时,B类没有 f(),则指向来自A类的f()函数。同时用基类的指针指向了派生类对象。最后析构函数的执 行。 六、程序设计题(本大题共1小题,共10分) 1. 已知交通工具类定义如下。 要求:(1)实现这个类;(2)定义并实现一个小车类car,是它的公有派生类,小车本身的私有 属性有载人数,小车的函数有init(设置车轮数,重量和载人数),getpassenger(获取载人数 ),print(打印车轮数,重量和载人数)。 class vehicle {protected: int wheels;//车轮数 float weight;//重量 public: void init(int wheels,float weight); int get_wheels(); float get_weight(); void print(); }; void vehicle::init(int wheels,float weight) {this->wheels=wheels; this->weight=weight; cout< int vehicle::get_wheels() {return wheels; } float vehicle::get_weight() {return weight;} void vehicle::print() {cout<<\车轮数:\重量:\答案:class car:public vehicle {private:int passengers; public: void init(int wheels,float weight,int pass); int getpassenger(); void print();}; void car::init(int wheels,float weight,int pass) {vehicle::init(wheels,weight); passengers=pass;} int car::getpassenger() {return passengers;} void car::print() {vehicle::print(); cout<<\可载人数:\}__ C++程序设计模拟试卷(四) 一、单项选择题(本大题共20小题,每小题1分,共20分)在每小题列出的四个备选项中 只有一个是符合题目要求的,请将其代码填写在题后的括号内。错选、多选或未选均无 分。 1. 当一个类的某个函数被说明为virtual时,该函数在该类的所有派生类中() A. 都是虚函数 B. 只有被重新说明时才是虚函数 C. 只有被重新说明为virtual时才是虚函数 D. 都不是虚函数 答案:A 解析:在基类声明为virtual的函数为虚函数,在派生类中只要有相同的函数(函数名相 同、返回值相同、形参类型和个数相同)即使不用virtual说明,也都是虚函数。 2. 要禁止修改指针p本身,又要禁止修改p所指向的数据,这样的指针应定义为() A. const char *p=“ABCD”; B. char *const p=“ABCD”; C. char const *p=“ABCD”; D. const char * const p=“ABCD”; 答案:D 解析: const char *p说明禁止通过p修改所指向的数据。char * const p则说明不能修改 指针p的地址。因此const char * const p=“ABCD”;它禁止修改指针p本身,又禁止修改p所指 向的数据。 3. 函数调用func((exp1,exp2),(exp3,exp4,exp5))中所含实参的个数为() A. 1 B. 2 C. 4 D. 5 答案:B 解析:(exp1,exp2)、(exp3,exp4,exp5)表示是两个逗号表达式,值是最后一个值,相当于两个 参数。因此实参的个数是2。 4. 设有函数模板 template 则下列语句中对该函数模板错误的使用是() A. Sum(10,2); B. Sum(5.0,6.7); C. Sum(15.2f,16.0f); D. Sum(“AB”,”CD”); 答案:D 解析:由Q Sum(Q x,Q y)可知形参和函数返回值都是同一种数据类型。A、B、C三项都正确 。而D项用字符串作为实参,字符串的操作与数值类型不同,要用特殊方法进行字符串的连接和 运算。 5. 类B是类A的公有派生类,类A和类B中都定义了虚函数func(),p是一个指向类A对象的指针 ,则p->A::func()将() A. 调用类A中的函数func() B. 调用类B中的函数func() C. 根据p所指的对象类型而确定调用类A中或类B中的函数func() D. 既调用类A中函数,也调用类B中的函数 答案:A 解析:指向类成员指针的使用,A::func()是明确调用A类的func函数,所以不管p指向基 类或者派生类对象,都执行基类虚函数。注意p->A::func()和p->fun();进行区分。如果使用p- >fun(),因为p指向派生类对象,由动态多态性可知要调用派生类的虚函数。 6. 在面向对象的程序设计中,首先在问题域中识别出若干个() A. 函数 B. 类 C. 文件 D. 过程 答案:B 解析:面向过程的和面向对象都具有、函数、文件和过程这些概念,而面向对象程序才有 类和对象的特征。所以选择B。 7. 已知f1和f2是同一类的两个成员函数,但f1不能直接调用f2,这说明() A. f1和f2都是静态函数 B. f1不是静态函数,f2是静态函数 C. f1是静态函数,f2不是静态函数 D. f1和f2都不是静态函数 答案:C 解析:普通成员函数可以调用静态函数,相反静态函数不能调用普通成员函数,这与普通 函数与常成员函数相同。因此选择C项。 8. 下列有关模板和继承的叙述正确的是() A. 模板和继承都可以派生出一个类系 B. 从类系的成员看,模板类系的成员比继承类系的成员较为稳定 C. 从动态性能看,继承类系比模板类系具有更多的动态特性 D. 相同类模板的不同实例一般没有联系,而派生类各种类之间有兄弟父子等关系 答案:D 解析:类是相同类型事物的抽象,具有不同的操作。而模板是不同类型的事物,具体相同 的操作的抽象。类模板的实例化后,各个对象没有任何关系。而类对象是通过派生、继承等关系 的关系。 9. 有关C++编译指令,以下叙述正确的是() A. C++每行可以写多条编译指令 B. #include指令中的文件名可含有路径信息 C. C++的编译指令可以以#或//开始 D. C++中不管#if后的常量表达式是否为真,该部分都需要编译 答案:B 解析:编译指令以#作为开头,只能一行写一条,#if有选择进行编译,所以选择B项。 10. 在C++中不返回任何类型的函数应该说明为() A. int B. char C. void D. double 答案:C 解析:无形参或无返回值都可以用void来声明,int char double分别是整型、字符型和实型。 11. 若Sample类中的一个成员函数说明如下: void set(Sample& a),则Sample& a的含义是() A. 指向类Sample的名为a的指针 B. a是类Sample的对象引用,用来作函数Set()的形参 C. 将a的地址赋给变量Set D. 变量Sample与a按位与的结果作为函数Set的参数 答案:B 解析:成员函数使用对象的引用作为形参。该函数的功能是将已知对象的所有数据成员的 值拷贝给相应对象的所有数据成员,不会建立临时对象,这里是对象的引用所以选择B。 12. 下列关于静态数据成员的描述中正确的是() A. 静态数据成员是类的所有对象所共有的 B. 静态数据成员要在构造函数内初始化 C. 类的每个对象有自己的静态数据成员 D. 静态数据成员不能通过类的对象调用 答案:D 解析:静态成员属于类的即所有对象所共享的,只能在外部进行初始化。使用时可以使用 形式有两种,类名::静态成员或者对象.静态成员。所以选择D项。 13. 在编译指令中,宏定义使用哪个指令() A. #if B. #include C. #define D. #error 答案:C 解析: #if条件编译,#include文件包含,#error错误处理。 14. 类的析构函数是对一个对象进行以下哪种操作时自动调用的是() A. 建立 B. 撤销 C. 赋值 D. 引用 答案:B 解析:删除对象或结束程序时,自动调用析构函数。 15. 关于new运算符的下列描述中,错误的是() A. 它可以用来动态创建对象和对象数组 B. 使用它创建的对象或对象数组可以使用运算符delete删除 C. 使用它创建对象时要调用构造函数 D. 使用它创建对象数组时必须指定初始值 答案:D 解析:静态成员的特性是静态成员只有一个拷贝(副本),这个副本被所有属于这个类的 对象共享。这种共享与全局变量或全局函数相比,既没有破坏数据隐藏的原则,又保证了安全性 。静态成员表示整个类范围的信息,其声明以关键字static开始,包括静态数据成员和静态成员 函数。在对静态数据成员初始化时应注意: (1)应在类体外对静态数据成员进行初始化(静态数据成员的初始化与它的访问控制权限无关 )。 (2)静态数据成员初始化时前面不加static关键字,以免与一般静态变量或对象混淆。 (3)由于静态数据成员是类的成员,因此在初始化时必须使用作用域运算符(::)限定它所属的 类。因此选择D项。 16. 如果类A被说明成类B的友元,则() A. 类B不一定是类A的友元 B. 类B的成员即类A的成员 C. 类A的成员即类B的成员 D. 类A的成员函数不得访问类B的成员 答案:A 解析:(P113)友元关系不能被继承,友元关系是单向的,友元关系不具有传递性。但是友元函数 不是类的成员,所以选择A项。 17. 假定一个类的构造函数为A(int aa,int bb){a=aa++;b=a*++bb;},则执行A x(4,5);语句 后,x.a和x.b的值分别为() A. 4和5 B. 4和20 C. 4和24 D. 20和5 答案:C 解析:执行构造函数将数据成员进行赋值,aa++是后加,先赋值a=4,++bb,bb变量值先自 加为6,再与a相乘,所以b=24。 18. 下列运算符中,在C++语言中不能重载的是() A. * B. >= C. :: D. / 答案:C 解析:除了类属关系运算符\、成员指针运算符\、作用域运算符\、sizeof运算 符和三目运算符\以外,C++中的所有运算符都可以重载。 19. C++语言建立类族是通过() A. 类的嵌套 B. 类的继承 C. 虚函数 D. 抽象类 答案:B 解析:类族即同一个类派生出来的类,各个类是兄弟或父子关系。 20. 在C++语言中,数据封装要解决的问题是() A. 数据的规范化 B. 便于数据转换 C. 避免数据丢失 D. 防止不同模块之间数据的非法访问 答案:D 解析:封装是指把对象属性和操作结合在一起,构成独立的单元,它的内部信息对外界是 隐蔽的,不允许外界直接存取对象的属性,只能通过有限的接口与对象发生联系。类是数据封装 的工具,对象是封装的实现。类的访问控制机制体现在类的成员中可以有公有成员、私有成员和 保护成员。对于外界而言,只需要知道对象所表现的外部行为,而不必了解内部实现细节。封装 体现了面向对象方法的“信息隐蔽和局部化原则”。 二、填空题(本大题共20小题,每小题1分,共20分)请在每小题的空格中填上正确答案 。错填、不填均无分。 1. 若要使用string类,需要引入的头文件是___。 答案: string.h [解析]编译时要将头文件引入才能使用标准库中的方法或成员。 2. 在函数前面用___保留字修饰时,则表示该函数表为内联函数。 答案: inline [解析]内联函数,用来提高程序运行速度。在类内部定义的函数也是内联函数。 3. 向量操作方法中___方法返回向量中的第一个对象。 答案: front [解析]考察向量方法的使用。front():返回向量中的第1个对象。back():返回向量中的最