计网OJ调了一万年,最后经ly同学提醒发现,问题是64位linux下,gcc编译时自动将我的int32类型补齐成了int64。OJ检测时又是通过直接读取内存的方法,导致测评结果一直只有50分。
我53分的OJ测评结果可供参考,具体如下图所示:
可以看出,stu_srv接收时全部错误,但计算结果部分正确,而且3轮检测,每轮都稳定地正确两次。当时特别疑惑。
具体情况如下:我的pdu结构体定义如下
struct rqt_pdu{
int32_t op;
int64_t op1;
int64_t op2;
}
不难看出,(32+64+64)/8=20,但是打印一下发现:
int main(){
struct rqt_pdu pdu;
printf("%ld\n", sizeof(pdu));
return 0;
}
这就是前面提到的,gcc编译自动将int32补齐成了int64。24=(64+64+64)/8
解决方法是加一句#pragma pack(1)
,指明结构体中的所有数据在内存中连续存储。再次打印:
特别注意一下:在使用
pragma pack (1)
之后,我们最好要unpack
. 否则在他之后定义的所有结构体都会按1字节对齐。我这里只有一个结构体,就不存在这个问题了。
结构体内存对齐具体可以看这篇博客,讲得很明白:https://blog.csdn.net/weixin_45157820/article/details/112755832