免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 4581 | 回复: 13
打印 上一主题 下一主题

[C] 关于多个相同全局变量在不同模块中定义的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2008-03-04 09:52 |只看该作者 |倒序浏览
/*test1.c*/
#include<stdio.h>
void f(void);

int x=15213;
int y=15212;

int main()
{
f();
printf("x=0x%x,y=0x%x\n",x,y);
return 0;
}
============================
/*test2.c*/
double x;

void f()
{
x=-0.0;
}
============================
在IA32/linux机器上运行结果如下:
x=0x0,y=0x80000000

有没有哪位高人能解释一下
非常感谢!!!

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
2 [报告]
发表于 2008-03-04 10:04 |只看该作者
错误的程序得到任何结果都不应该感到奇怪。

论坛徽章:
0
3 [报告]
发表于 2008-03-04 10:16 |只看该作者
up....

这个问题好像很复杂

  1. //a.h
  2. int x;
复制代码

  1. //a.c
  2. #include "a.h"
复制代码

  1. //main.c
  2. #include "a.h"
  3. int main(){
  4. x=43;
  5. return 0;
  6. }
复制代码

编译的时候

  1. gcc a.c shared -o a
  2. gcc main.c -o main
复制代码

main显式加在共享库a,然后程序里边的x变量怎么处理,main函数中修改的x到底是谁。如果我想只有一个x,要怎样做?

论坛徽章:
0
4 [报告]
发表于 2008-03-04 10:31 |只看该作者
这个是链接器如何解释多处定义的全局变量的问题
因为double型为8个字节,而int型为4字节
所以x=-0.0将用负数的双精度浮点表示覆盖存储器中x和y的值
现在有个疑问:
既然是覆盖但是为什么x为0x0,而y为0x80000000呢
运行环境是模拟的linux(cygwin中)


顺便问一下
在vc6.0中为什么double型和float型的数以十六进制(比如printf(“%x”,x))输出时无论数是多少,总是显示为0x0呢
是编译器不支持浮点数的IEEE浮点表示规则么
还是要另外配置?

论坛徽章:
0
5 [报告]
发表于 2008-03-04 10:34 |只看该作者
/usr/bin/ld: Warning: alignment 4 of symbol `x' in /tmp/cc26QdVn.o is smaller than 8 in /tmp/ccBqiV91.o

绝不能放过一个Warning.编译时把-Wall加上吧.
编译器行为跟了一下,没想通,等高人指点.

论坛徽章:
0
6 [报告]
发表于 2008-03-04 12:40 |只看该作者
这段代码能运行么?我怀疑

论坛徽章:
0
7 [报告]
发表于 2008-03-04 12:49 |只看该作者
见《深入理解计算机系统》 链接器一章。 关于符号重复定义的问题。

int x = 15213; 即是声明,又是定义,所以是强符号。
double x;  没有初值,不一定是定义,是弱符号。
所以你的程序里面,最终链接到程序里面的是int x, 占4个字节。

而在test2.c中,因为有double x的声明。 所以程序以为x是一个8字节的double数, 所以对x操作时,会写到x本身4字节空间以外的地址,碰巧你的y在那个位置,所以就变成了80000000 。

8不就是你的符号位,-0;

[ 本帖最后由 agaric 于 2008-3-4 12:52 编辑 ]

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
8 [报告]
发表于 2008-03-04 12:50 |只看该作者
原帖由 agaric 于 2008-3-4 12:49 发表
见《深入理解计算机系统》 链接器一章。 关于符号重复定义的问题。

int x = 15213; 即是声明,又是定义,所以是强符号。
double x;  没有初值,不一定是定义,是弱符号。
所以你的程序里面,最终链接到程 ...

说的好!

论坛徽章:
0
9 [报告]
发表于 2008-03-04 12:53 |只看该作者
另外我不记得printf 可以用 %x 来打印浮点数呀。

论坛徽章:
3
2015年迎新春徽章
日期:2015-03-04 09:56:11数据库技术版块每日发帖之星
日期:2016-08-03 06:20:00数据库技术版块每日发帖之星
日期:2016-08-04 06:20:00
10 [报告]
发表于 2008-03-04 13:28 |只看该作者
错误的程序得到了错误的结果
如果你真想搞懂怎么回事情,那么你就去想想double是怎么存储的吧,如果你不懂双精度是怎么存储的,那么去查吧
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP