Intel和微软同时出现的C语言面试题 nnhI]#,a{
#pragma pack(8) \h[*oeh
RU/WI<O
struct s1{ =g6~2p=H
short a; yD\Kn{
long b; /NB|N*}O)
}; pFMJG<W9,
sE]z.Po=
struct s2{ N68]r3/K
char c; mW)kWuOO
s1 d; 3BK
8{/
long long e; x2fqfrr_]
}; /Cwwz
f8K0/z
#pragma pack() =@O&$&
%Qj$@.*:
问 l-r$czY
1.sizeof(s2) = ? ,]JIp~=nsh
2.s2的s1中的a后面空了几个字节接着是b? J0bcW25
L}Sb0 o.
)/!HI0TU
hyPS 6Y'1
如果您知道答案请在讨论中写出,以下是部份网友的答案,供参考: (5`(H.(
A]QGaWK
网友rwxybh(行云)的答案: D
dwFKc&
内存布局是 *>aVU'
1*** 11** @ukL!AV?Y
1111 **** -h|[8UG^b
1111 1111 |4BD
'%e@7Cs
所以答案就是24和3 )Dv;,t
|2TH[J_a
下面是一个测试的程序,试一试就知道了,我用的是VC2005 j."V>p8u$
KJ&I4CU]^
#pragma pack(8) j-aTpN
4+>~Ui_#
struct s1{ pIrL7Pb0
short a; // 2 BYtes Q+a&a]*KL^
long b; // 4 Bytes !+Cc^{
}; TG?>;It&
struct s2{ R'F \9eyA
char c; // 1 Byte ?^:5`
s1 d; // 8 Bytes }|/<!l+;$
long long e; // 8 Bytes e
GAto
}; \{*`-Pv
// 1*** 11** g|^U?|;p
// 1111 **** TRgj`FG
// 1111 1111 ;x_T*} CH
// t#f-3zd9
w"kBAi&
// 00 01 02 03 04 05 06 07 X/%!p<}:'
// 00 01 02 03 04 05 06 07 :zIB3nT^
// 00 01 02 03 04 05 06 07 JC$_Pg!
// g]MgT-C|
#pragma pack() (:H4
M?sTz@tqq
int main(int argc, char* argv[]) .pxUO3g
{ R'_F9\
s2 a; m/g[9Y
char *p = (char *)&a; mm!JNb9(
for(int i=0;i<24;++i) ;)f,A)(Z
p = (char)(i%8); asvM/ 9
printf("%d\n",sizeof(a)); 3# 0Nd"/0
printf("c=0x%lx\n",a.c); u&`rK7J
printf("d.a=0x%x\n",a.d.a); OWr\$lm@z$
printf("d.b=0x%x\n",a.d.b); d@ZXCiA},
printf("e=0x%llx\n",a.e); H2g#'SK@
return 0; {P?p*2J'
} k'"R;^~xg
结果: W>CG;x{
24 !*qQ7
c=0x0 n|.>41bJ
d.a=0x504 6~!7?FK
d.b=0x3020100 KCa @0
e=0x706050403020100 um".Z4S
yJ; ;&
#K-O<:s=y
网友 redleaves (ID最吊的网友)的答案和分析: {v d+cE
A)SnPbI-p
如果代码: _!Z}HCk
#pragma pack(8) qpf|.m
struct S1{ G!F_Q7|-
char a; Z_jV0[\v0P
long b; %gqu7}'
}; Ql}#mC.>/
struct S2 { ?5 6;<%0
char c; s<C66z
struct S1 d; p)Ht =~
long long e; <pT1p4T<
}; Y!u">M#@
#pragma pack() dqt}:^L*0g
sizeof(S2)结果为24. }p9#Bzc
成员对齐有一个重要的条件,即每个成员分别对齐.即每个成员按自己的方式对齐. ZD?LsD 3
也就是说上面虽然指定了按8字节对齐,但并不是所有的成员都是以8字节对齐.其对齐的规则是,每个成员按其类型的对齐参数(通常是这个类型的大小)和指定对齐参数(这里是8字节)中较小的一个对齐.并且结构的长度必须为所用过的所有对齐参数的整数倍,不够就补空字节. zU|'IW&
TuwSJS7
S1中,成员a是1字节默认按1字节对齐,指定对齐参数为8,这两个值中取1,a按1字节对齐;成员b是4个字节,默认是按4字节对齐,这时就按4字节对齐,所以sizeof(S1)应该为8; ZQ\O|
n8
S2中,c和S1中的a一样,按1字节对齐,而d 是个结构,它是8个字节,它按什么对齐呢?对于结构来说,它的默认对齐方式就是它的所有成员使用的对齐参数中最大的一个,S1的就是4.所以,成员d就是按4字节对齐.成员e是8个字节,它是默认按8字节对齐,和指定的一样,所以它对到8字节的边界上,这时,已经使用了12个字节了,所以又添加了4个字节的空,从第16个字节开始放置成员e.这时,长度为24,已经可以被8(成员e按8字节对齐)整除.这样,一共使用了24个字节. Z2]\k|%<Fa
a b ZOJ7^g
S1的内存布局:11**,1111, q+4<"b+6G
c S1.a S1.b d XS0NjZW
S2的内存布局:1***,11**,1111,****11111111 Q;z'"P
>O1u![9K|w
这里有三点很重要: ,I f9w$(z
1.每个成员分别按自己的方式对齐,并能最小化长度 W\ARCcTQ
2.复杂类型(如结构)的默认对齐方式是它最长的成员的对齐方式,这样在成员是复杂类型时,可以最小化长度 ))6iVgSE$
3.对齐后的长度必须是成员中最大的对齐参数的整数倍,这样在处理数组时可以保证每一项都边界对齐 kQ6YQsJ.*
J<iiA:&J
gyMy;}a
网友xue23(xue23) 的答案和分析: i~DLo3
V8%( h[
有程序查一下各个变量的内存地址得知: Zqg
AgN@
各个变量在内存中的位置为 bwjLMWEVq
c***aa** _(@ezX.p
bbbb**** b]Lp_t
dddddddd :7qJ[k{g
测试代码为: >hotkMX `3
s2 ss; }"^d<dvuz
cout << "ss.c = " << &ss << endl ; ~X) 1!Sr
cout << "ss.d.a = " <<&ss.d.a << endl; K;g6V!U
cout << "ss.d.b = " <<&(ss.d.b) < cout << "ss.d = " <<&ss.e << endl; b:*(
f#"q
print out 各个变量的内存地址不就可以看出来了吗。 f1Gyl
gEq";B%?
所以答案是24,2. l2
#^}-
h4>q~&Pd
但是我的想像中应该是这样的分布情况: Y-"7R>^I
c******* q+67Wc=
aa**bbbb `v Ebm Xb
dddddddd .uo:fxbd2
&nn+X%m9g
不知为什么会c和a放在一起,组成8位长度。