Chinaunix

标题: 关于多个相同全局变量在不同模块中定义的问题 [打印本页]

作者: joe_zh128    时间: 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

有没有哪位高人能解释一下
非常感谢!!!
作者: flw    时间: 2008-03-04 10:04
错误的程序得到任何结果都不应该感到奇怪。
作者: reiase    时间: 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,要怎样做?
作者: joe_zh128    时间: 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浮点表示规则么
还是要另外配置?
作者: RobinsonNie    时间: 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加上吧.
编译器行为跟了一下,没想通,等高人指点.
作者: yuangong    时间: 2008-03-04 12:40
这段代码能运行么?我怀疑
作者: agaric    时间: 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 编辑 ]
作者: flw    时间: 2008-03-04 12:50
原帖由 agaric 于 2008-3-4 12:49 发表
见《深入理解计算机系统》 链接器一章。 关于符号重复定义的问题。

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

说的好!
作者: agaric    时间: 2008-03-04 12:53
另外我不记得printf 可以用 %x 来打印浮点数呀。
作者: cjaizss    时间: 2008-03-04 13:28
错误的程序得到了错误的结果
如果你真想搞懂怎么回事情,那么你就去想想double是怎么存储的吧,如果你不懂双精度是怎么存储的,那么去查吧
作者: flw    时间: 2008-03-04 13:31
原帖由 cjaizss 于 2008-3-4 13:28 发表
错误的程序得到了错误的结果
如果你真想搞懂怎么回事情,那么你就去想想double是怎么存储的吧,如果你不懂双精度是怎么存储的,那么去查吧

admire
作者: joe_zh128    时间: 2008-03-04 15:14
谢谢各位!
我知道这个程序会得到错误的结果,只不过想知道为什么会出错,怎么出错的而已

To:agaric
我就是在你说的那本书上看到的,因为不解,所以发上来
-0.0即是0x80000000
但为什么x中没有存储,全都存到y中了呢
作者: joe_zh128    时间: 2008-03-04 15:47
understand!

多谢!
作者: 想飞的蜗牛    时间: 2008-03-04 22:33

  1. main()
  2. {
  3.     int i;
  4.     char c;
  5.     for(i = 0; i< 5; i++){
  6.     scanf("%d",&c);
  7.     printf("%d ",i);
  8.     }
  9.     printf("\n");
  10. }
复制代码


跟《c陷阱与缺陷》一书这个例子有点异曲同工之妙
不知道理解错没有

[ 本帖最后由 想飞的蜗牛 于 2008-3-4 22:36 编辑 ]




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2