您的位置:首页 > 其它

深入理解sizeof和strlen

2011-07-08 15:02 405 查看
首先看看sizeof和strlen在MSDN上的定义:

首先看一MSDN上如何对sizeof进行定义的:
sizeofOperatorsizeofexpressionThesizeofkeywordgivestheamountofstorage,inbytes,associatedwithavariableoratype(includingaggregatetypes).Thiskeywordreturnsavalueoftypesize_t.Theexpressioniseitheranidentifieroratype-castexpression_r(atypespecifierenclosedinparentheses).Whenappliedtoastructuretypeorvariable,sizeofreturnstheactualsize,whichmayincludepaddingbytesinsertedforalignment.Whenappliedtoastaticallydimensionedarray,sizeofreturnsthesizeoftheentirearray.Thesizeofoperatorcannotreturnthesizeofdynamicallyallocatedarraysorexternalarrays.

然后再看一下对strlen是如何定义的:
strlenGetthelengthofastring.RoutineRequiredHeader:strlen<string.h>size_tstrlen(constchar*string);Parameterstring:Null-terminatedstringLibrariesAllversionsoftheCrun-timelibraries.ReturnValueEachofthesefunctionsreturnsthenumberofcharactersinstring,excludingtheterminalNULL.Noreturnvalueisreservedtoindicateanerror.RemarksEachofthesefunctionsreturnsthenumberofcharactersinstring,notincludingtheterminatingnullcharacter.wcslenisawide-characterversionofstrlen;theargumentofwcslenisawide-characterstring.wcslenandstrlenbehaveidenticallyotherwise.

二、由几个例子说开去。

第一个例子:
char*ss="0123456789";
sizeof(ss)结果4===》ss是指向字符串常量的字符指针
sizeof(*ss)结果1===》*ss是第一个字符
charss[]="0123456789";
sizeof(ss)结果11===》ss是数组,计算到\0位置,因此是10+1
sizeof(*ss)结果1===》*ss是第一个字符
charss[100]="0123456789";
sizeof(ss)结果是100===》ss表示在内存中的大小100×1s
trlen(ss)结果是10===》strlen是个函数内部实现是用一个循环计算到\0为止之前
intss[100]="0123456789";
sizeof(ss)结果400===》ss表示再内存中的大小100×4
strlen(ss)错误===》strlen的参数只能是char*且必须是以''\0''结尾的
charq[]="abc";charp[]="a\n";
sizeof(q),sizeof(p),strlen(q),strlen(p);结果是4332
第二个例子:
classX{inti;intj;chark;};
Xx;cout<<sizeof(X)<<endl;结果12===》内存补齐
cout<<sizeof(x)<<endl;结果12同上
第三个例子:
charszPath[MAX_PATH]
  如果在函数内这样定义,那么sizeof(szPath)将会是MAX_PATH,但是将szPath作为虚参声明时(voidfun(charszPath[MAX_PATH])),sizeof(szPath)却会是4(指针大小)

三、sizeof深入理解。1.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned int类型。该类型保证能容纳实现所建立的最大对象的字节大小。
2.sizeof是算符,strlen是函数。
3.sizeof可以用类型做参数,strlen只能用char*做参数,且必须是以''\0''结尾的。sizeof还可以用函数做参数,比如:
shortf();printf("%d\n",sizeof(f()));
输出的结果是sizeof(short),即2。
4.数组做sizeof的参数不退化,传递给strlen就退化为指针了。
5.大部分编译程序在编译的时候就把sizeof计算过了是类型或是变量的长度这就是sizeof(x)可以用来定义数组维数的原因
charstr[20]="0123456789";inta=strlen(str);//a=10;intb=sizeof(str);//而b=20;

6.strlen的结果要在运行的时候才能计算出来,时用来计算字符串的长度,不是类型占内存的大小。
7.sizeof后如果是类型必须加括弧,如果是变量名可以不加括弧。这是因为sizeof是个操作符不是个函数。
8.当适用了于一个结构类型时或变量,sizeof返回实际的大小,当适用一静态地空间数组,sizeof归还全部数组的尺寸。sizeof操作符不能返回动态地被分派了的数组或外部的数组的尺寸
9.数组作为参数传给函数时传的是指针而不是数组,传递的是数组的首地址,如:
fun(char[8])fun(char[])
都等价于fun(char*)在C++里传递数组永远都是传递指向数组首元素的指针,编译器不知道数组的大小如果想在函数内知道数组的大小,需要这样做:进入函数后用memcpy拷贝出来,长度由另一个形参传进去
fun(unsigedchar*p1,intlen){unsignedchar*buf=newunsignedchar[len+1]memcpy(buf,p1,len);}
有关内容见:C++PRIMER?
10.计算结构变量的大小就必须讨论数据对齐问题。为了CPU存取的速度最快(这同CPU取数操作有关,详细的介绍可以参考一些计算机原理方面的书),C++在处理数据时经常把结构变量中的成员的大小按照4或8的倍数计算,这就叫数据对齐(dataalignment)。这样做可能会浪费一些内存,但理论上速度快了。当然这样的设置会在读写一些别的应用程序生成的数据文件或交换数据时带来不便。MSVC++中的对齐设定,有时候sizeof得到的与实际不等。一般在VC++中加上#pragmapack(n)的设定即可.或者如果要按字节存储,而不进行数据对齐,可以在Options对话框中修改Advancedcompiler页中的Dataalignment为按字节对齐。
11.sizeof操作符不能用于函数类型,不完全类型或位字段。不完全类型指具有未知存储大小的数据类型,如未知存储大小的数组类型、未知内容的结构或联合类型、void类型等。如sizeof(max)若此时变量max定义为int max(),sizeof(char_v) 若此时char_v定义为char char_v [MAX]且MAX未知,sizeof(void)都不是正确形式
四、结束语

sizeof使用场合。1.sizeof操作符的一个主要用途是与存储分配和I/O系统那样的例程进行通信。例如: 
  void *malloc(size_t size), size_t fread(void * ptr,size_t size,size_t nmemb,FILE * stream)。

2.用它可以看看一类型的对象在内存中所占的单元字节。
void * memset(void * s,int c,sizeof(s))

3.在动态分配一对象时,可以让系统知道要分配多少内存。
4.便于一些类型的扩充,在windows中就有很多结构内型就有一个专用的字段是用来放该类型的字节大小。
5.由于操作数的字节数在实现时可能出现变化,建议在涉及到操作数字节大小时用sizeof来代替常量计算。
6.如果操作数是函数中的数组形参或函数类型的形参,sizeof给出其指针的大小。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: