Intel和微软同时出现的C语言面试题 ES40?o*]x
#pragma pack(8) V\6]n2
\}]!)}G
struct s1{ O`vTnrY
short a; Zkf0p9h\
long b; DfKr[cqLM
};
`7H4Y&E
]n-:Yv5 W
struct s2{ 9Vf1Xz
char c; qpXWi
&g
s1 d; (dv]=5""
long long e; a5w:u5
}; ":_vK}5
2=_gf
#pragma pack() f47M#UC
zhf.NCSt(
问 O eL}EVs8=
1.sizeof(s2) = ? Bm]8m=p
2.s2的s1中的a后面空了几个字节接着是b? c*@G_rb
QD%L0;j
<^$<#Kd
rl0< Ls
如果您知道答案请在讨论中写出,以下是部份网友的答案,供参考: 8.[SU
'e6WDC1Am(
网友rwxybh(行云)的答案: GQ
|Mr{.;
内存布局是 t#2(j1
1*** 11** XU"~h64]
1111 **** {GJ@psG*
1111 1111 k?'B*L_Mzv
?Ae ven
所以答案就是24和3 u7=U^}#
[}&Sxgv
下面是一个测试的程序,试一试就知道了,我用的是VC2005 >KJ+-QuO&
) Yd?m0m*
#pragma pack(8) r\/+Oa'
M|Rb&6O
struct s1{ x*/S*!vx\
short a; // 2 BYtes oJfr +3I
long b; // 4 Bytes F;]%V%F.X
}; Phke`3tth
struct s2{ @*sWu_-Y%
char c; // 1 Byte =%/)m:f!^
s1 d; // 8 Bytes YIjTL!bA"
long long e; // 8 Bytes nvPwngEQm
}; q`r**N+zn
// 1*** 11** f&
CBU
// 1111 **** 8w.YYo8`
// 1111 1111 RU\/j%^
// =AuR:Tx
k1!@^A
// 00 01 02 03 04 05 06 07 cb}[S:&|
// 00 01 02 03 04 05 06 07 uS^Ipxe\
// 00 01 02 03 04 05 06 07 yeMB0Z*r
// ZMq6/G*fD
#pragma pack() s)pbS}L
Sm5H_m!
int main(int argc, char* argv[]) v\{!THCSh
{ vuYSVI2=H
s2 a; O6OP =K!t:
char *p = (char *)&a; F|!){=
for(int i=0;i<24;++i) 1@-Ns
p = (char)(i%8); <%"b9T`'
printf("%d\n",sizeof(a)); hq #?kN
printf("c=0x%lx\n",a.c); \o^2y.q:>
printf("d.a=0x%x\n",a.d.a); j*vYBGD
printf("d.b=0x%x\n",a.d.b); #Q
/Arq
printf("e=0x%llx\n",a.e); =y-@AU8
return 0; 4H/fP]u
} GI1
结果: 5@BBoeG
24 ?[lV-
c=0x0 <.? jc%
d.a=0x504 q*>&^V $M
d.b=0x3020100 H/37)&$E(
e=0x706050403020100 J_4!2v!6e
FIsyiSY<j
>j QWn@
网友 redleaves (ID最吊的网友)的答案和分析: J7g8D{4
v'3J.?N
如果代码: .yEBOMNZ
#pragma pack(8) 7yh/BZ1
struct S1{ @qYp>|AF
char a; Uw7h=UQh
long b; ~
(jKz}'~U
}; T]c%!&^_
struct S2 { lx7Q.su'
char c; &:`U&06q
struct S1 d; Kuu *&u
long long e; WA&!;Zq
}; #NryLE!/
#pragma pack() _+E5T*dk
sizeof(S2)结果为24. ilqy/fL#
成员对齐有一个重要的条件,即每个成员分别对齐.即每个成员按自己的方式对齐. qO|R^De
也就是说上面虽然指定了按8字节对齐,但并不是所有的成员都是以8字节对齐.其对齐的规则是,每个成员按其类型的对齐参数(通常是这个类型的大小)和指定对齐参数(这里是8字节)中较小的一个对齐.并且结构的长度必须为所用过的所有对齐参数的整数倍,不够就补空字节. m* kl
1bn^.768l
S1中,成员a是1字节默认按1字节对齐,指定对齐参数为8,这两个值中取1,a按1字节对齐;成员b是4个字节,默认是按4字节对齐,这时就按4字节对齐,所以sizeof(S1)应该为8; =UfsL%
S2中,c和S1中的a一样,按1字节对齐,而d 是个结构,它是8个字节,它按什么对齐呢?对于结构来说,它的默认对齐方式就是它的所有成员使用的对齐参数中最大的一个,S1的就是4.所以,成员d就是按4字节对齐.成员e是8个字节,它是默认按8字节对齐,和指定的一样,所以它对到8字节的边界上,这时,已经使用了12个字节了,所以又添加了4个字节的空,从第16个字节开始放置成员e.这时,长度为24,已经可以被8(成员e按8字节对齐)整除.这样,一共使用了24个字节. XSyHk"g`
a b m+T;O/lG0{
S1的内存布局:11**,1111, e0,|Wm
c S1.a S1.b d q}?4f*WC
S2的内存布局:1***,11**,1111,****11111111 ys kO
Z'7
这里有三点很重要: %Da1(bBh
1.每个成员分别按自己的方式对齐,并能最小化长度 WL"^>[Vq
2.复杂类型(如结构)的默认对齐方式是它最长的成员的对齐方式,这样在成员是复杂类型时,可以最小化长度 jr:7?8cH0L
3.对齐后的长度必须是成员中最大的对齐参数的整数倍,这样在处理数组时可以保证每一项都边界对齐 _y}
T/I9
bl&nhI)w
P&^;656r
网友xue23(xue23) 的答案和分析: wLnf@&jQ%
yL-YzF2
有程序查一下各个变量的内存地址得知: G\+L~t
各个变量在内存中的位置为 |M,iM]
c***aa** QvKh,rBFVG
bbbb**** 7V!*NBsl
dddddddd )u`[6,d
测试代码为: `M^=
D&Bf
s2 ss; y1+*6|
cout << "ss.c = " << &ss << endl ; z?*w8kU&>
cout << "ss.d.a = " <<&ss.d.a << endl; 7\ s"o&G
cout << "ss.d.b = " <<&(ss.d.b) < cout << "ss.d = " <<&ss.e << endl; ?b>,9A.Z
print out 各个变量的内存地址不就可以看出来了吗。 2OVRf0.R~
)x=1]T>v"'
所以答案是24,2. =E#%'/ A;c
2KYw}j|5
但是我的想像中应该是这样的分布情况: S(*sw
0O@+
c******* +Z!)^j
aa**bbbb .Z
`av n
dddddddd x#xFh0CA
:Ra,Eu
不知为什么会c和a放在一起,组成8位长度。