2007年下半年程序员下午试卷


全国计算机技术与软件专业技术资格(水平)考试


2007年下半年程序员下午试卷

试题一(共15分)

阅读以下说明和流程图,填补流程图中的空缺(1)~(5),将解答填入答题纸的对应栏内。

[说明]

某单位动态收集的数据中常包含重复的数据,所以需要进行处理,使得重复的数据仅出现一次。下面流程图的功能是:在nn1)个数据D1D2Dn中,选出其中所有不重复的k个数据,置于原来前k个数据的位置上。

该流程图的算法如下:第1个数据必然被选出,然后从第2个数据开始,逐个考察其余的数据。假设D1D2Dmm1)是已经选出的、不重复的数据,则对于数据Dim<i≤n),将其依次与DmDm-1D1进行比较,若没有发现与之相同者,则Di被选出并置于Dm+1的位置上;否则对Di不做处理。

例如,如下10个数据:

5227447191    n=10

经过上述算法处理后的结果为:

527419                k=6

[流程图]

注:循环开始的说明按照“循环变量名:循环初值,循环终值,增量”格式描述。

试题二(共15分)

阅读以下说明和C语言函数,将应填入n处的字句写在答题纸的对应栏内。

[说明]

已知190011日是星期一,下面的函数count_5_13(int year)用于计算给定的年份year中有几个“黑色星期五”。“黑色星期五”指既是13日又是星期五的日期。

函数count_5_13(int year)首先算出年份year113日是星期几,然后依次计算每个月的13日是星期几,若是星期五,则计数。

程序中使用了函数isLeapYear(int year),其功能是判断给定年份是否为闰年,返回值为1(0)分别表示year是(或不是)闰年。

[C语言函数]

intcount_5_13(int year)

{

  int date;        /* date0表示星期日,为16分别表示星期一至星期六 */

  long days = 0;   /* days记录天数*/

int m, y,c = 0; /* c用于表示黑色星期五的个数 */


  if (year < 1900)     return -1;


/*计算从190011日起,至给定年份year113日间隔的天数*/

days =12;

for (y =1900; y < year; y++) {

  days += 365;

  if (isLeapYear(y))  1;

  }

  date = ((days % 7) + 1) % 7;  /* 算出给定年份year113日是星期几 */

  c = (2) ? 1 : 0;

 for(m = 1;  3; m++) {

             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; break;

                case 2: days = 28;

                                  if (4)    days = 29;

                            break;

             }/* end of switch*/

       date =((days % 7) + 5) % 7;

             if (date == 5)        c++;

} /* endof for*/

  return c;

}

试题三(共15分)

阅读以下说明和C语言程序,将应填入n处的字句写在答题纸的对应栏内。

[说明]

某电信公司记录了每个用户的详细通话情况(每次通话数据记录在一行),现将某用户某月的通话数据存入一个文本文件“dial.txt”,其数据格式如下:

拨入或拨出标记通话开始时间通话结束时间对方号码

  注1:数据字段以一个空格作为分隔符。

2:拨入和拨出标记均为小写字母。拨入标记为“i”,表示其他用户呼叫本机,本机用户不需付费;拨出标记为“o”,表示本机呼叫其他用户,此时本机用户需要付费。

3:通话开始和结束时间的格式均为:HH:MM:SS。其中HH表示小时,取值0023MM表示分钟,取值0059SS表示秒,取值0059。从通话开始到结束这段时间称为通话时间,假定每次通话时间以秒为单位,最短为1秒,最长不超过24小时。

4:跨月的通话记录计入下个月的通话数据文件。

  例如“o 23:01:12 00:12:15 …”表示本次通话是本机呼叫其他用户,时间从230112秒至次日的01215秒,通话时间为7103秒。

下面程序的功能是计算并输出该用户本月电话费(单位:)。  

通话计费规则为:

1. 月通话费按每次通话费累加;

2. 每次的通话费按通话时间每分钟0.08元计算,不足1分钟时按1分钟计费。

对于每次的拨出通话,程序中先分别计算出通话开始和结束时间相对于当日000秒的时间长度(以秒为单位),然后算出本次通话时间和通话费。

例如,若输入文件dial.txt的数据如下所示,则输出fee = 7.44

o14:05:23 14:11:25 82346789

i15:10:00 16:01:15 13890000000

o10:53:12 11:07:05 63000123

o 23:01:1200:12:15 13356789001


[C程序代码]

#include<stdio.h>


FILE*fin;

intmain()

{

  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)  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) /* 若通话开始和结束时间跨日 */

               interval = 3 -t_start + t_end;

        else

               interval = t_end - t_start;


c = 4;    /* 计算完整分钟数表示的通话时间 */

        if (interval % 60)

5;

        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


链表的结点类型定义如下:

typedefstruct Node {

  int data;

  struct Node *next;

}NODE;


[C语言函数]

void compress(NODE*head)

{  NODE *ptr,*q;

  ptr = 1;   /* 取得第一个元素结点的指针 */

  while (2 && ptr -> next)  {

       q = ptr -> next;

       while(q && 3) {  /* 处理重复元素 */

4 = q -> next;

           free(q);

           q = ptr -> next;

       }

5 = ptr -> next;

}/* endof while */

}/* endof compress */



从下列3道试题(试题五至试题七)中任选1道解答。如果解答的试题数超过1道,则题号小的1道解答有效。



试题五(共15分)

阅读下列说明、图和C++代码,回答问题1至问题3,将解答写在答题纸的对应栏内。

[说明]

已知四个类之间的关系如图5-1所示,分别对每个类的方法进行编号,例如Shapeperimeter()方法为1号,表示为“1:perimeter()”,Rectangle类的perimeter()2号,表示为“2:perimeter()”,依此类推,其中,每个类的perimeter方法都为虚函数且方法签名相同。














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()        1

      sh->perimeter()     2

      sq->height()        3

      sq->perimeter()     4

      sh->height()        5

      tr->perimeter()     6

[问题3] 不考虑内存释放问题,下列赋值语句中哪两个是合法的(写出合法赋值语句的编号)。

sq = sh;  sh = tr;  tr = sq;  sq = tr;  sh = sq;

试题六(共15分)

阅读以下应用说明以及Visual Basic程序代码,将应填入n处的字句写在答题纸的对应栏内。

[应用说明]

某电视台拟开发应用程序来显示戏曲大赛中14号四位选手决赛的现场投票情况。该程序从现场观众中(不超过2000人)每0.5秒收集一次对这四位选手的支持票数,并在屏幕上动态显示这四位选手的票柱(以高度反映票数)与累计得票数,如图6-1所示。投票过程限时30秒,每名观众最多为1名选手投票。投票结束后系统通过比较各位选手的累计得票数,显示决赛结果:“*号胜出”(如有单个冠军)或“继续进行PK”(如有多人获得相同的最高票数)。

6-1

在开发该程序的过程中创建的主要控件及其初始属性值说明如下:

控件名

类型

用途

初始属性设置

ShpM1 to 4

形状数组

显示各选手得票情况

矩形,实心,高度0

txtM(1  to 4)

文本框数组

显示各选手得票数

0

cmdStart

命令按钮

启动计票

标题:开始投票

txtResult

文本框

显示决赛结果

Tim1

计时器

每半秒收集处理一次

间隔时间0.5秒,关闭状态

该程序中设置公共变量T动态记录投票时间。四个形状ShpM1 to 4)动态增长的底线固定。

[VisualBasic程序代码]

Dim T AsInteger      '声明全局变量

PrivateSub Form_Load()

  For i = 1 To 4

   ShpM(i).Top = 2000 :  ShpM(i).Height = 0   ' 初始票柱高度为0

   TxtM(i).Text = 0

 Next i

 Tim1.Enabled= False : Tim1.Interval = 500 : T = 0

End Sub


PrivateSub CmdStart_Click()

  Tim1.Enabled = True                   '开始投票

  CmdStart.Enabled = False

End Sub


PrivateSub Tim1_Timer()

 Dim n(1 To 4) As Integer              ' n(1 to 4)为每次收集的票数

Dim i AsInteger, j As Integer  

Dim G AsInteger                      ' G用于计算最高票数

  Dim ngAs Integer                     ' ng用于计算冠军个数

 For i = 1 To 4

   n(i) = …                            ' 收集i号选手的票数,此处省略

   TxtM(i).Text = TxtM(i).Text + n(i)   ' 累计票(VB能进行自动转换)

   ShpM(i).Top = ShpM(i).Top - n(i)

   ShpM(i).Height = ShpM(i).Height + 1   ' 算出票柱高度

 Next i

 T = T + 1                                  ' 计时

 If T = 60 Then                             ' 投票时间到

2                                ' 停止数据收集处理

   ng = 1                                

   G = TxtM(1).Text                      

   For i= 2 To 4

      If G< TxtM(i).Text Then

         G = TxtM(i).Text  

ng = 3

j = i

      Else

         If G = TxtM(i).Text  Then  ng= 4    ' 计算冠军个数

      End If

   Nexti

   If ng = 1 Then

     txtResult.Text = 5                       ' 单个冠军结果

   Else

     txtResult.Text = "继续进行PK"      

   EndIf

  End If

End Sub


试题七(共15分)

阅读下列说明、图和Java代码,回答问题1至问题3,将解答写在答题纸的对应栏内。

[说明]

已知四个类之间的关系如图7-1所示,分别对每个类的方法进行编号,例如Shapeperimeter()1号,表示为“1:perimeter()”,Rectangle类的perimeter()2号,表示为“2:perimeter()”,依此类推,其中,每个类的perimeter方法签名相同。















7-1 类图

[Java 代码]

Triangletr = new Triangle();

Square sq= new Square();

Shape  sh = tr;


[问题1] 关于上述Java代码中sh tr的以下叙述中,哪两个是正确的(写出编号)。

sh tr分别引用同一个对象;

sh tr分别引用同一类型的不同的对象;

sh tr分别引用不同类型的不同对象;

sh tr分别引用同一个对象的不同拷贝;

sh tr所引用的内存空间是相同的。

[问题2] 写出下面消息对应的方法编号(如果该消息错误或者没有对应的方法调用,请填写“无”)。

      tr.height()        1

      sh.perimeter()     2

      sq.height()        3

      sq.perimeter()     4

      sh.height()        5

      tr.perimeter()     6

[问题3] 下列赋值语句中哪两个是合法的(写出合法赋值语句的编号)。

sq = sh;  sh = tr;  tr = sq;  sq = tr;  sh = sq;


你可能感兴趣的:(程序员考试)