Chinaunix首页 | 论坛 | 博客
  • 博客访问: 282288
  • 博文数量: 28
  • 博客积分: 690
  • 博客等级: 上士
  • 技术积分: 358
  • 用 户 组: 普通用户
  • 注册时间: 2011-08-25 20:39
文章分类

全部博文(28)

文章存档

2012年(12)

2011年(16)

分类:

2011-08-26 19:24:51

/***************************************************

操作系统:WIN7

CPU : Intel i3

编译环境: C-FREE 5.0

编译器: C89

******************************************************/

#include

#include

char string[2000];
int main()

int index;
int *p;
p=string;

 for(index=0;index<500;index++)  
{
  p[index] = -1 - index;
 } 
   printf("strlen(string) = %d\n",strlen(string)); 
return 1;

}

运行结果:

    一开始,我看到这段代码的时候,就认为运行结果为2000. 当实际运行之后,嘿嘿,竟然是上图所示,很意外。跟网友讨论了一番,才明白其来龙去脉以及此段代码考察的知识点,当然,也发现自身的基础不是一般的差,需要多充电去复习那些最原始的东东。

    废话就不多说了,下面就分析一下这段代码:

    char string[2000];    //定义了一个长度为2000的字符型数组string,且默认情况下,其每个元素的                           //初值均为0x00.  

     int index;         //定义32位整型数据index
     int *p;            //定义指向32位整型数据的指针P;
     p=string;          //把string的初始地址赋给p,即p和string均指向相同的地址

    for(index=0;index<500;index++)    //循环500次,给p指向的500个整数赋值,地址从p 至 p+499
    {
        p[index] = -1 - index;
    }  

      printf("strlen(string) = %d\n",strlen(string));  //计算并输出string中第一次出现0x00之前                                                        //的数据个数。  

    如果对strlen()的特点和用法非常熟悉,那么这段代码对你来说已经懂了一半,抓着string中什么时候出现0x00这个点不放,去查找string中的元素。根据上面的for循环,可以得知代码通过给p中的元素赋值来间接地对string中的元素赋值,p中的一个元素就相当于string在内存中对应地址的的连续四个字节,由于intel的CPU支持的是小端模式,所以p[0] = -1在内存中的存放形式为 0xff 0xff 0xff 0xff(内存地址由低到高排列),p[1] = -2   在内存中的存放形式为 0xfe 0xff 0xff 0xff  ,。。。。,p[127] = -128 在内存中的存放形式为0x10 0xff 0xff 0xff ,  p[128] = -129 在内存中的存放形式为 0x7f 0xff 0xff 0xff, 。。。, p[255] = -256 在内存中的存放形式为 0x00 0xff 0xff 0xff, 到此,出现了一个0x00,所以当执行strlen时,在string中的第(255*4+1)个元素处停止,因此,输出为255*4 = 1020。为了更直观的看到前面所述的情况特在代码中添加了打印信息,如下所示:

int main()
{
char dd = 128;
int index;
int *p;
p=string;
for(index=0;index<500;index++)
{
p[index] = -1-500;
}
 printf("string[0*4] = %d\n",string[0*4]);//index = 0时,p[0]的低字节
 printf("string[127*4] = %d\n",string[127*4]);//index=127,p[127]的低字节
printf("string[128*4] = %d\n",string[128*4]);
//index=128,p[128]的低字节
printf("string[129*4] = %d\n",string[129*4]);
//index=129,p[129]的低字节
printf("string[254*4] = %d\n",string[254*4]);
//index=254,p[254]的低字节
printf("string[255*4] = %d\n",string[255*4]);
//index=255,p[255]的低字节
printf("strlen(string) = %d\n",strlen(string));
return 1;
}

运行结果如下:



总结了一下,这段简短的代码大概包含了下面四个知识点:

(1)不同数据类型的长度以及取值范围;

(2)数据在内存中的存放形式;

(3)不同CPU平台的数据存放模式;

(4)函数strlen()的特点。


思考一下:

当CPU支持大端模式时,那么输出会是多少呢?

当p的元素值为正值时,假如从按照1+index的形式进行变化,输出结果又是多少?


下面再给两个有点变化的代码:

(1)

#include
#include
char string[2001];
int main()
{
char dd = 128;
int index;
int *p;
p=string;
for(index=0;index<500;index++)
{
p[index] = -1;
}
/*
printf("string[0*4] = %d\n",string[0*4]);
printf("string[127*4] = %d\n",string[127*4]);
printf("string[128*4] = %d\n",string[128*4]);
printf("string[129*4] = %d\n",string[129*4]);
printf("string[254*4] = %d\n",string[254*4]);
 printf("string[255*4] = %d\n",string[255*4]); */

printf("strlen(string) = %d\n",strlen(string)); 
return 1;
}

(2)

#include
#include
char string[2000];
int main()
{
char dd = 128;
int index;
int *p;
p=string;
for(index=0;index<500;index++)
{
p[index] = -1;
}

/*
printf("string[0*4] = %d\n",string[0*4]);
printf("string[127*4] = %d\n",string[127*4]);
printf("string[128*4] = %d\n",string[128*4]);
printf("string[129*4] = %d\n",string[129*4]);
printf("string[254*4] = %d\n",string[254*4]);
 printf("string[255*4] = %d\n",string[255*4]); */

printf("strlen(string) = %d\n",strlen(string)); 

return 1;
}

二者的运行结果相同么?为什么?

很显然,不一定相同,第一种情况输出肯定为2000,第二种的不确定,因为超出数组string的边界了,边界外的数值不确定是多少,因为不确定何时停止。

阅读(977) | 评论(0) | 转发(0) |
0

上一篇:没有了

下一篇:Linux Kernel with NULL Pointer Dereference

给主人留下些什么吧!~~