重解几道华为经典C语言面试题

重解几道华为经典C语言面试题 面试


         1、找错 数组

    一、找错 函数

  void test1()
{
    char string[10];
    char* str1="0123456789";
    strcpy(string, str1);
}
spa


 
    这里string数组越界,由于字符串长度为10,还有一个结束符‘\0’。因此总共有11个字符长度。string数组大小为10,这里越界了。

内存

    PS:使用strcpy函数的时候必定要注意前面目的数组的大小必定要大于后面字符串的大小,不然即是访问越界。 ci

  void test2()
{
    char string[10], str1[10];
    for(i=0; i<10;i++)
    {
       str1[i] =’a’;
    }
    strcpy(string, str1);
}
字符串


 
    这里有一个一眼就能看出的问题,那就是变量i没有定义,这在代码编译阶段编译器能够帮你发现,很容易搞定。然而不少问题是本身形成的漏洞,编译器是帮不上什么忙的。这里最大的问题仍是str1没有结束符,由于strcpy的第二个参数应该是一个字符串常量。该函数就是利用判断第二个参数的结束符来获得是否拷贝完毕。因此在for循环后面应加上str1p[9] = ‘\0’

get

    PS:字符数组和字符串的最明显的区别就是字符串会被默认的加上结束符‘\0’ 编译器

  void test3(char* str1)
{
    char string[10];
    if(strlen(str1)<=10)
    {
       strcpy(string, str1);
    }
}
string


 
    这里的问题还是越界问题。strlen函数获得字符串除结束符外的长度。若是这里是<=10话,就很明显越界了。

    小结:上面的三个找错的函数,主要是考查对字符串和字符数组的概念的掌握以及对strcpy函数和strlen函数的理解。

    二、找错

  DSN get_SRM_no()
{
  static int SRM_no;
  int I;
  for(I=0;I<MAX_SRM;I++)
    {
       SRM_no %= MAX_SRM;
       if(MY_SRM.state==IDLE)
       {
         break;
       }
    }
    if(I>=MAX_SRM)
       return (NULL_SRM);
    else
       return SRM_no;
}


 
    这里for循环的判断语句是后来我加上的,估计在网上流传的时候被人给弄丢了,根据对程序的分析,给补上了。估计错误应该不是这儿。

    简单的阅读一下这个函数,能够大概的能够猜想出这个函数的功能是分配一个空闲的SRAM块。方法:从上次分配的RAM块后的RAM块开始检测SRAM每一个RAM块,看是不是IDLE状态,若是是IDLE则返回当前的RAM块的号SRM_no.若是全部的RAM块都不是IDLE状态,则意味着没法分配一个RAM给函数调用者,返回一个表示没有RAM可分配的标志(NULL_SRM)。

    通过上面的分析,则这里能够知道,这个函数的错误是for循环里面没有给SRM_no这个变量累加1.

    三、写出程序运行结果

  int sum(int a)
{
    auto int c=0;
    static int b=3;
    c+=1;
    b+=2;
    return(a+b+c);
}
void main()
{
    int I;
    int a=2;
    for(I=0;I<5;I++)
    {
       printf("%d,", sum(a));
    }
}


    运行结果是:810121416

    在求和函数sum里面cauto变量,根据auto变量特性知每次调用sum函数时变量c都会自动赋值为0.bstatic变量,根据static变量特性知每次调用sum函数时变量b都会使用上次调用sum函数时b保存的值。

    简单的分析一下函数,能够知道,若传入的参数不变,则每次调用sum函数返回的结果,都比上次多2.因此答案是:810121416

  4func(1) = ?
int func(int a)
{
    int b;
    switch(a)
    {
        case 1: 30;
        case 2: 20;
        case 3: 16;
        default: 0;
    }
    return b;
}


    在 case 语句中可能忘记了对变量b赋值。若是改成下面的代码:

  int func(int a)
{
    int b;
    switch(a)
    {
        case 1:      b = 30;
        case 2:      b = 20;
        case 3:      b = 16;
        default:     b = 0;
    }
    return b;
}


 
    由于case语句中漏掉了break语句,因此不管传给函数的参数是多少,运行结果均为0.

    五、a[q - p] = ?

   int a[3]

    a[0]=0 a[1]=1 a[2]=2

    int *p *q

    p=a

    q=&a[2]

    很明显:a[q - p] = a[2] = 2


    六、内存空间占用问题

    定义 int **a[3][4], 则变量占有的内存空间为:16位系统2432位编译系统中是48.

    PS:公式:3 * 4 * sizeofint ** .

    七、程序编写

    编写一个函数,要求输入年月日时分秒,输出该年月日时分秒的下一秒。如输入20041231235959秒,则输出200511000秒。

void ResetTheTime(int *year,int *month,int *date,int *hour,int *minute,int*second)
{
    int dayOfMonth[12]={31,28,31,30,31,30,31,31,30,31,30,31};

    if( *year < 0   || *month < 1 || *month > 12 ||        *date < 1   || *date > 31 || *hour < 0   || *hour > 23 ||        *minute < 0 ||*minute > 59|| *second <0 || *second >60 )        return;     if( *year%400 == 0 || *year%100 != 0 && *year%4 == 0 )        dayOfMonth[1] = 29;     if(*second >= 60)     {        *second = 0;        *minute += 1;        if(*minute >= 60)        {            *minute = 0;            *hour += 1;            if(*hour >= 24)            {               *hour = 0;               *date += 1;               if(*date > dayOfMonth[*month-1])               {                   *date = 1;                   *month += 1;                   if(*month > 12)                   {                      *month=1;                       *year += 1;                   }               }            }        }     }     return; }

相关文章
相关标签/搜索