C/C++面试笔试(2)

1.

#include <stdio.h>

#include <setjmp.h>

#include<string.h>

#include <conio.h>

#include <stdlib.h>

static jmp_buf buf;//

int main(void)

{

   volatile int b = 3;//用来修饰被不同线程访问和修改的变量

   if (setjmp(buf) != 0)

   {

     printf("%d\n", b);

     exit(0);

  }

  b = 5;

  longjmp(buf, 1);//发生异常跳转

}


输出结果:5


详细解析:

关键点在于理解setjmp以及longjmp,第一次运行到setjmp,会设置jmp_buf,然后返回0。当调用longjmp时,会把longjmp里面的非0值作为setjmp的返回值返回(如果longjmp的value参数为0,setjmp恢复后返回1,也就是当恢复到setjmp存储点的时候,setjmp一定不会返回0)。

setjmp和logjmp是配合使用的,用它们可以实现跳转的功能,和goto语句很类似,不同的是goto只能实现在同一个函数之内的跳转,而setjmp和logjmp可以实现在不同函数间的跳转

用法:

首先用setjmp设置跳转的地点,setjmp的参数buf是用来保存设置跳转点时的函数使用的重要数据,当从其他函数跳转回来,如果不用这个保存的数据恢复当前函数的一些数据的话,跳转回来是不能运行的。第一次设置的时候setjmp返回值为0

使用longjmp就可以跳转到setjmp的地方了,参数buf就是使用setjmp的时候保存的,而第二个参数会在跳转以后把这个值让setjmp返回的,也就是longjmp第二个参数,就是跳转到setjmp之后setjmp函数要返回的值

 

这个代码里运行步骤

1.先执行setjmp,因为是第一次设置跳转点,返回值是0,不执行if语句块里的语句,

2.然后执行b=5,b的值就是5了

3.再执行longjmp跳转之后, 最后再执行setjmp, 这时setjmp会返回1(也就是longjmp的第二个参数指定的值),就会执行if语句块里的语句----打印之后终止程序,这时b的值是5,就会打印出5来



2.

#include<stdio.h>

int main(void)

{

   struct node

   {

      int a;

      int b;

      int c;

   };

   struct node s = { 3, 5, 6 };

   struct node *pt = &s;

   printf("%d\n", *(int*)pt);

   return 0;

}

输出结果:3


详细解析:

(int *)pt指向结构体的第一个成员的地址,pt指向结构体,(int*)pt把pt强制类型转换为指针类型,*(int*)pt就是取指针pt指向的值。

pt是一个指向某类型的指针,*pt就是pt指向元素的值,*pt=x就是把那个值赋值给x。


拓展:

struct MyStruct
{
char dda; //偏移量为0,满足对其方式,dda占用1个字节。
double dda1;  /* 可用的地址偏移量为1,不是sizeof(double)的倍数,为满足对齐方式需要补足7个字节才能使偏移量变为8,vc自动填充7个字节,dda1存放                  在偏移量为8的地址上,它占用8个字节。*/
int type /*下一个可用地址的偏移量为16,是sizeof(int)的倍数,满足int的对其方式,所以不需要vc自动填充,type存放在偏移量为16的地址上,他占用4字节*/
}

 /*所有成员变量都分配了空间,空间总的大小为1+7+8+4=20,不是结构的节边界数——结构中占用最大空间的类型所占用的字节数,即sizeof(double)=8的倍数。所以该结构总的大小为:sizeof(MyStruct)为1+7+8+4+4=24.其中总的有7+4=11个字节是vc自动填充的,没有放任何有意义的东西*/


练习(1).class B

{

private:

bool m_bTemp;

int m_nTemp;

bool m_bTemp2;

        };

cout<<sizeof(B)<<endl;

答案:12   (注意bool 1字节)


 练习(2).class A

{

double d;

float a;

int b;

static int b; 

char c;

A();

~A();

};

A a;

  cout<<sizeof(A)<<endl;

 cout<<sizeof(a)<<endl;

答案:24  24(静态变量时存放在全局数据区的,而sizeof计算栈中分配的大小,是不会计算在内的)


练习(3)以下代码输出是什么?

class B

{

float f;

char p;

int adf[3];

};

cout<<""<<sizeof(B);

答案:20

解析:

偏移量 内存
0 4
4 1
5+3 12

最终结果:sizeof(B)=4+1+12+3=20

4.

int foo(int x, int n){

   int val = 1;

   if (n > 0)

   {

       if (n % 2 == 1) val *= x;

       val *= foo(x * x, n / 2);

   }

   return val;

}


输出结果:xn

详细解析:

foo(x, 0) = 1
foo(x, 1) = x * foo(x, 0) = x
foo(x, 2) = foo(x * x, 1) = x * x
foo(x, 3) = x * foo(x, 2) = x * (x * x)


5.1

#include <stdio.h>

int main(void)

{

  int a[5] = { 1, 2, 3, 4, 5 };

  int *ptr = (int*)(&a + 1);

  printf("%d %d\n", *(a + 1), *(ptr - 1));

  return 0;

}

输出结果:2 5

详细解析:

表达式 结果
*a 1
*(a+1) 2
*(a+2) 3
*(x+i) ==x[i]

&a是一个隐式的指向int[5]数组的指针,它和int* ptr是不一样的,如果真要定义这个指针,应该是int(*ptr)[5]。所以ptr每一次加一操作都相当于跨越int[5]的内存步长(也就是5个int长度),也就是说&a+ 1其实就是指向了a[5]这个位置,实际上内存里面这个位置是非法的,但是对ptr的强制转换导致了后面ptr-1的内存步长改为了1个int长度,所以ptr-1实际指向了a[4]。


5.2

#include <stdio.h>

int main(void)

{

  int a[][3] = {1, 2, 3, 4, 5, 6};

  int (*ptr)[3] = a;

  printf("%d %d ", (*ptr)[1], (*ptr)[2]);

  ++ptr;

  printf("%d %d\n", (*ptr)[1], (*ptr)[2]);

  return 0;

}


输出结果:2 3 5 6

 详细解析:

ptr为指向int[3]数组的指针,首先指向a[0],所以(*ptr)[1],(*ptr)[2]就是a[0][1], a[0][2].然后++ptr,相当于ptr指向了a[1],这时得到的是a[1][1]a[1][2],所以结果就是23,5, 6


5.3

#include <stdio.h>

int main(void)

{

  char p;

  char buf[10] = {1, 2, 3, 4, 5, 6, 9, 8};

  p = (buf + 1)[5];

  printf("%d\n", p);

  return 0;

}

输出结果:9

详细解析:

记住,x[i]相当于*(x + i),所以(buf + 1)[5]等价于*(buf + 1 + 5),或buf[6]。






你可能感兴趣的:(面试,C语言)