c语言常见面试题(二)

本文介绍了多个C语言面试中常见的题目,包括检测处理器字节序、计算序列和、大数阶乘、宏定义的使用、字符串循环右移、static和const关键字的作用,以及程序设计相关的问题。每个题目都有详细的解答和分析,帮助读者理解C语言的关键概念和技巧。
摘要由CSDN通过智能技术生成

试题1:请写一个C函数,若处理器是Big_endian的,则返回0;若是Little_endian的,则返回1

解答:

int checkCPU()
{
 {
  union w
  {
   int a;
   char b;
  } c;
  c.a = 1;
  return (c.b == 1);
 }
}

剖析:

   嵌入式系统开发者应该对Little-endian和Big-endian模式非常了解。采用Little-endian模式的CPU对操作数的存放方 式是从低字节到高字节,而Big-endian模式对操作数的存放方式是从高字节到低字节。例如,16bit宽的数0x1234在Little- endian模式CPU内存中的存放方式(假设从地址0x4000开始存放)为:

内存地址 存放内容
0x4000 0x34
0x4001 0x12

  而在Big-endian模式CPU内存中的存放方式则为:

内存地址 存放内容
0x4000 0x12
0x4001 0x34

  32bit宽的数0x123456在Little-endian模式CPU内存中的存放方式(假设从地址0x4000开始存放)为:

内存地址 存放内容
0x4000 0x56
0x4001 0x34
0x4002 0x12
0x4003 0x00

  而在Big-endian模式CPU内存中的存放方式则为:

内存地址 存放内容
0x4000 0x00
0x4001 0x12
0x4002 0x34
0x4003 0x56

  联合体union的存放顺序是所有成员都从低地址开始存放,面试者的解答利用该特性,轻松地获得了CPU对内存采用Little-endian还是Big-endian模式读写

 

试题2:写一个函数返回1+2+3+…+n的值(假定结果不会超过长整型变量的范围)

解答:


 

int Sum(int n)
{
  return ( (long)1 + n) * n/ 2;  //或return (1l + n) * n / 2;

}

剖析:
 
对于这个题,只能说,也许最简单的答案就是最好的答案。下面的解答,或者基于下面的解答思路去优化,不管怎么“折腾”,其效率也不可能与直接return ( 1 l + n ) * n / 2相比!

自动类型转换:

 如果一个运算符两边的运算数类型不同,先要将其转换为相同的类型,即较低类型转换为较高类型,然后再参加运算,转换规则如下图所示。
double ←── float 高

long

unsigned

int ←── char,short 低
● 图中横向箭头表示必须的转换,如两个float型数参加运算,虽然它们类型相同,但仍要先转成double型再进行运算,结果亦为double型。 纵向箭头表示当运算符两边的运算数为不同类型时的转换,如一个long 型数据与一个int型数据一起运算,需要先将int型数据转换为long型, 然后两者再进行运算,结果为long型。所有这些转换都是由系统自动进行的, 使用时你只需从中了解结果的类型即可。

试题3:不用第三变量交换两个数
 

voidswap(int a,int b)
{
a=a^b;
    b=b^a;
    a=a^b;
}
或者
void swap(int a,int b)
{
    a=a+b;
    b=a-b;
    a=a-b;
}
不过这两种方法只是修改了函数的形参,如果要修改实参,可以采用如下的方法:
void swap(int* a,int*b)
{
    *a=*a^*b;
    *b=*b^*a;
    *a=*a^*b;
    printf("In %s:a=%d,b=%d\n",__FUNCTION__,*a,*b);
}

试题4:求大数的阶乘例如100!,使用通常的做法会溢出,这里要使用数组的方法。

 #include <stdio.h>

int len,a[201];  /*数字长度与存数字的数组,其中a[1]是个位,a[len]是最高位*/

void mul(int x)  /*乘法过程*/
{
  int i;
  for (i=1;i<=len;i++) a[i]*=x;  /*一位一位的乘*/
  for (i=1;i<=len+1;i++)  /*进位*/
  {
    a[i+1]+=a[i]/10;
    a[i]%=10;
  }
  len+=2;   /*计算新的数字长度*/
  while (a[len]==0) len--;  
}

void out()  /*输出数字*/
{
  int i;
  for (i=len;i>0;i--) printf("%d",a[i]);
  printf("\n");
}

int main()
{
  int i;
  for (i=0;i<=200;i++) a[i]=0;  /*初始化赋*/
  len=1;
  a[1]=1;
  for (i=2;i<=100;i++) mul(i);  /*从2到100高精乘*/
  out();  /*输出*/
  return 0;
}

试题5:写一个“标准”宏MIN,这个宏输入两个参数并返回较小的一个。另外,当你写下面的代码时会发生什么事?

least = MIN(*p++, b);

解答:

#define MIN(A,B) ((A) <= (B) ? (A) : (B))
MIN(*p++, b)会产生宏的副作用

剖析:

这个面试题主要考查面试者对宏定义的使用,宏定义可以实现类似于函数的功能,但是它终归不是函数,而宏定义中括弧中的“参数”也不是真的参数,在宏展开的时候对“参数”进行的是一对一的替换。程序员对宏定义的使用要非常小心,特别要注意两个问题:

(1) 谨慎地将宏定义中的“参数”和整个宏用用括弧括起来。所以,严格地讲,下述解答:

#define MIN(A,B) (A) <= (B) ? (A) : (B)
#define MIN(A,B) (A <= B ? A : B )  都应判0分;

(2) 防止宏的副作用。

  宏定义#define MIN(A,B) ((A) <= (B) ? (A) : (B))对MIN(*p++, b)的作用结果是:((*p++) <= (b) ? (*p++) : (*p++)) 这个表达式会产生副作用,指针p会作三次++自增操作。除此之外,另一个应该判0分的解答是:#define MIN(A,B) ((A) <= (B) ? (A) : (B));
这个解答在宏定义的后面加“;”,显示编写者对宏的概念模糊不清,只能被无情地判0分并被面试官淘汰。

 

试题6:为什么标准头文件都有类似以下的结构?
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值