logo资料库

2007下半年程序员考试真题及答案-下午卷.doc

第1页 / 共10页
第2页 / 共10页
第3页 / 共10页
第4页 / 共10页
第5页 / 共10页
第6页 / 共10页
第7页 / 共10页
第8页 / 共10页
资料共10页,剩余部分请下载后查看
2007 下半年程序员考试真题及答案-下午卷 试题一(共 15 分) 阅读以下说明和流程图,填补流程图中的空缺(1)~(5),将解答填入答题纸的对应 栏内。 [说明] 某单位动态收集的数据中常包含重复的数据,所以需要进行处理,使得重复的数据仅 出现一次。下面流程图的功能是:在 n(n≥1)个数据 D1、D2、…、Dn 中,选出其中所有不 重复的 k 个数据,置于原来前 k 个数据的位置上。 该流程图的算法如下:第 1 个数据必然被选出,然后从第 2 个数据开始,逐个考察其 余的数据。假设 D1、D2、…、Dm(m≥1)是已经选出的、不重复的数据,则对于数据 Di(m
试题二(共 15 分) 阅读以下说明和 C 语言函数,将应填入 (n) 处的字句写在答题纸的对应栏内。 [说明] 已知 1900 年 1 月 1 日是星期一,下面的函数 count_5_13(int year)用于计算给定的年 份 year 中有几个“黑色星期五”。“黑色星期五”指既是 13 日又是星期五的日期。 函数 count_5_13(int year)首先算出年份 year 的 1 月 13 日是星期几,然后依次计算 每个月的 13 日是星期几,若是星期五,则计数。 程序中使用了函数 isLeapYear(int year),其功能是判断给定年份是否为闰年,返回 值为 1(或 0)分别表示 year 是(或不是)闰年。 [C 语言函数] int count_5_13(int year) { int date; long days = 0; int m, y, c = 0; /* c 用于表示黑色星期五的个数 */ /* date 为 0 表示星期日,为 1~6 分别表示星期一至星期六 */ /* days 记录天数 */ if (year < 1900) return -1; /*计算从 1900 年 1 月 1 日起,至给定年份 year 的 1 月 13 日间隔的天数*/ days = 12; for (y = 1900; y < year; y++) { days += 365; if (isLeapYear(y)) (1)days++,days+=1,days=days+1 ; } date = ((days % 7) + 1) % 7; c = ( (2)date==5) ? 1 : 0; for(m = 1; (3)m < 12; m++) /* 算出给定年份 year 的 1 月 13 日是星期几 */ { switch (m) { case 1: case 3: case 5: case 7: case 8: case 10: case 12: days = 31; break; case 4: case 6: case 9: case 11: days = 30; case 2: days = 28; break; if ( (4)isLeapYear(year) 或 year%4==0 && year%100!=0 || year%400==0 ) days = 29; break; }/* end of switch*/ date =((days % 7) + (5)date if (date == 5) c++; ) % 7; } /* end of for*/ return c; }
试题三(共 15 分) 阅读以下说明和 C 语言程序,将应填入 (n) 处的字句写在答题纸的对应栏内。 [说明] 某电信公司记录了每个用户的详细通话情况(每次通话数据记录在一行),现将某用户 某月的通话数据存入一个文本文件“dial.txt”,其数据格式如下: 拨入或拨出标记 通话开始时间 通话结束时间 对方号码 注 1:数据字段以一个空格作为分隔符。 注 2:拨入和拨出标记均为小写字母。拨入标记为“i”,表示其他用户呼叫本机,本 机用户不需付费;拨出标记为“o”,表示本机呼叫其他用户,此时本机用户需要付费。 注 3:通话开始和结束时间的格式均为:HH:MM:SS。其中 HH 表示小时,取值 00~23; MM 表示分钟,取值 00~59;SS 表示秒,取值 00~59。从通话开始到结束这段时间称为通话 时间,假定每次通话时间以秒为单位,最短为 1 秒,最长不超过 24 小时。 注 4:跨月的通话记录计入下个月的通话数据文件。 例如“o 23:01:12 00:12:15 …”表示本次通话是本机呼叫其他用户,时间从 23 时 01 分 12 秒至次日的 0 时 12 分 15 秒,通话时间为 71 分 03 秒。 下面程序的功能是计算并输出该用户本月电话费(单位:元)。 通话计费规则为: 1. 月通话费按每次通话费累加; 2. 每次的通话费按通话时间每分钟 0.08 元计算,不足 1 分钟时按 1 分钟计费。 对于每次的拨出通话,程序中先分别计算出通话开始和结束时间相对于当日 0 点 0 分 0 秒的时间长度(以秒为单位),然后算出本次通话时间和通话费。 例如,若输入文件 dial.txt 的数据如下所示,则输出 fee = 7.44。 o 14:05:23 14:11:25 82346789 i 15:10:00 16:01:15 13890000000 o 10:53:12 11:07:05 63000123 o 23:01:12 00:12:15 13356789001 [C 程序代码] #include FILE *fin; int main() { char str[80]; int h1,h2,m1,m2,s1,s2; long t_start,t_end, interval; int c; double fee = 0; fin = fopen("dial.txt","r"); if (!fin) return -1; while (!feof(fin)) {
if (!fgets(str,80,fin)) break; if ( (1)str[0] == 'i',或*str == 'i' ) continue; h1 = (str[2] - 48) * 10 + str[3] - 48; m1 = (str[5] - 48) * 10 + str[6] - 48; s1 = (str[8] - 48) * 10 + str[9] - 48; h2 = (str[11] - 48) * 10 + str[12] - 48; m2 = (str[14] - 48) * 10 + str[15] - 48; s2 = (str[17] - 48) * 10 + str[18] - 48; t_start = h1*60*60 + m1*60 + s1; /* 通话开始时间 */ t_end = h2*60*60 + m2*60 + s2; /* 通话结束时间 */ if ( (2)t_end < t_start ) /* 若通话开始和结束时间跨日 */ interval = (3)24*60*60 - t_start + t_end; else interval = t_end - t_start; c = (4)interval / 60 if (interval % 60) ; /* 计算完整分钟数表示的通话时间 */ (5)c++,c+=1,c=c+1 ; } fee += c * 0.08; } fclose(fin); printf("fee = %.2lf\n",fee); return 0;
试题四(共 15 分) 阅读以下说明和 C 语言函数,将应填入 (n) 处的字句写在答题纸的对应栏内。 [说明] 已知包含头结点(不存储元素)的单链表的元素已经按照非递减方式排序,函数 compress(NODE *head)的功能是去掉其中重复的元素,使得链表中的元素互不相同。 处理过程中,当元素重复出现时,保留元素第一次出现所在的结点。 图 4-1(a)、(b)是经函数 compress()处理前后的链表结构示例图。 图 4-1 链表的结点类型定义如下: typedef struct Node { int data; struct Node *next; }NODE; [C 语言函数] void compress(NODE *head) { NODE *ptr,*q; ptr = (1)head -> next ; while ( (2)ptr && ptr -> next) { /* 取得第一个元素结点的指针 */ q = ptr -> next; while(q && (3)q->data == ptr->data 或 ptr->next->data == ptr->data ) { /* 处理重复元素 */ (4)ptr -> next = q -> next; free(q); q = ptr -> next; } (5) ptr = ptr -> next; }/* end of while */ }/* end of compress */
从下列 3 道试题(试题五至试题七)中任选 1 道解答。如 果解答的试题数超过 1 道,则题号小的 1 道解答有效。 试题五(共 15 分) 阅读下列说明、图和 C++代码,回答问题 1 至问题 3,将解答写在答题纸的对应栏内。 [说明] 已知四个类之间的关系如图 5-1 所示,分别对每个类的方法进行编号,例如 Shape 的 perimeter()方法为 1 号,表示为“1:perimeter()”,Rectangle 类的 perimeter()为 2 号, 表示为“2:perimeter()”,依此类推,其中,每个类的 perimeter 方法都为虚函数且方法签 名相同。 Shape 1:perimeter() 2:perimeter() Rectangle Triangle 3:height() 5:perimeter() 4:perimeter() Square 图 5-1 类图 [C++代码] Triangle *tr = new Triangle(); Square *sq = new Square(); Shape *sh = tr; [问题 1] 关于上述 C++代码中 sh 和 tr 的以下叙述中,哪两个是正确的(① ⑤)。 ① sh 和 tr 分别引用同一个对象; ② sh 和 tr 分别引用同一类型的不同的对象; ③ sh 和 tr 分别引用不同类型的不同对象; ④ sh 和 tr 分别引用同一个对象的不同拷贝; ⑤ sh 和 tr 所引用的内存空间是相同的。 [问题 2] 写出下面消息对应的方法编号(如果该消息错误或者没有对应的方法调用,请填 写“无”)。 tr->height() sh->perimeter() sq->height() sq->perimeter() sh->height() tr->perimeter() (1) 3 (2) 5 (3) 无 (4) 4 (5)无 (6) 5 [问题 3] 不考虑内存释放问题,下列赋值语句中哪两个是合法的(写出合法赋值语句的② ⑤)。 ① sq = sh; ② sh = tr; ③ tr = sq; ④ sq = tr; ⑤ sh = sq;
试题六(共 15 分) 阅读以下应用说明以及 Visual Basic 程序代码,将应填入 (n) 处的字句写在答题纸 的对应栏内。 [应用说明] 某电视台拟开发应用程序来显示戏曲大赛中 1~4 号四位选手决赛的现场投票情况。该 程序从现场观众中(不超过 2000 人)每 0.5 秒收集一次对这四位选手的支持票数,并在屏 幕上动态显示这四位选手的票柱(以高度反映票数)与累计得票数,如图 6-1 所示。投票过 程限时 30 秒,每名观众最多为 1 名选手投票。投票结束后系统通过比较各位选手的累计得 票数,显示决赛结果:“*号胜出”(如有单个冠军)或“继续进行 PK”(如有多人获得相同 的最高票数)。 图 6-1 在开发该程序的过程中创建的主要控件及其初始属性值说明如下: 控件名 类型 用途 初始属性设置 ShpM(1 to 4) 形状数组 显示各选手得票情况 矩形,实心,高度 0 txtM(1 to 4) 文本框数组 显示各选手得票数 0 cmdStart txtResult Tim1 命令按钮 启动计票 标题:开始投票 文本框 计时器 显示决赛结果 空 每半秒收集处理一次 间隔时间 0.5 秒,关闭状态 该程序中设置公共变量 T 动态记录投票时间。四个形状 ShpM(1 to 4)动态增长的底 线固定。 [Visual Basic 程序代码] Dim T As Integer Private Sub Form_Load() '声明全局变量 For i = 1 To 4 ShpM(i).Top = 2000 : TxtM(i).Text = 0 ShpM(i).Height = 0 ' 初始票柱高度为 0 Next i Tim1.Enabled = False : Tim1.Interval = 500 : T = 0 End Sub
Private Sub CmdStart_Click() Tim1.Enabled = True CmdStart.Enabled = False End Sub Private Sub Tim1_Timer() Dim n(1 To 4) As Integer Dim i As Integer, j As Integer Dim G As Integer Dim ng As Integer For i = 1 To 4 '开始投票 ' n(1 to 4)为每次收集的票数 ' G 用于计算最高票数 ' ng 用于计算冠军个数 n(i) = … TxtM(i).Text = TxtM(i).Text + n(i) ShpM(i).Top = ShpM(i).Top - n(i) ShpM(i).Height = ShpM(i).Height + (1)n(i) ' 收集 i 号选手的票数,此处省略 ' 累计票(VB 能进行自动转换) ' 算出票柱高度 Next i T = T + 1 If T = 60 Then ' 计时 ' 投票时间到 (2)Tim1.Enabled = False ' 停止数据收集 处理 ng = 1 G = TxtM(1).Text For i = 2 To 4 If G < TxtM(i).Text Then G = TxtM(i).Text ng = (3)1 j = i Else If G = TxtM(i).Text Then ng = (4)ng + 1 ' 计算冠军个数 End If Next i If ng = 1 Then txtResult.Text = (5)Str$(j) + "号胜出",或 Str(j) + "号胜出" ' 单个冠军结果 Else txtResult.Text = "继续进行 PK" End If End If End Sub
分享到:
收藏