88bifa必发唯一官网 1

整到一块,Sizeof与Strlen的差距与沟通

写在头里

一、sizeof
sizeof(…)是运算符,在头文件中typedef为unsigned
int,其值在编写翻译时即计算好了,参数能够是数组、指针、类型、对象、函数等。
它的作用是:获得保证能兼容完毕所树立的最大目的的字节大小。
是因为在编写翻译时总结,由此sizeof不能用来回到动态分配的内部存款和储蓄器空间的分寸。实际上,用sizeof来回到类型以及静态分配的目的、结构或数组所占的上空,再次来到值跟对象、结构、数组所蕴藏的内容并没有涉嫌。
具体来讲,当参数分别如下时,sizeof再次回到的值表示的意思如下:
数组——编写翻译时分配的数组空间尺寸;
指南针——存款和储蓄该指针所用的长台湾空中大学小(存款和储蓄该指针的地点的长短,是长整型,应该为四);
花色——该品种所占的长空尺寸;
对象——对象的实际占领空间尺寸;
函数——函数的回到类型所占的半空中尺寸。函数的归来类型不能够是void。

一、sizeof
    sizeof(…)是运算符,而不是3个函数。
    二个轻巧易行的例证:
int a;
cout<<sizeof a<<endl;
    在头文件中typedef为unsigned
int,其值在编写翻译时即总括好了,参数能够是数组、指针、类型、对象、函数等。
    它的效益是:获得保障能包容达成所确立的最大目的的字节大小
 
  出于在编写翻译时总结,由此sizeof不能够用来回到动态分配的内部存款和储蓄器空间的轻重。
   
实际上,用sizeof来回到类型以及静态分配的对象、结构或数组所占的长空,重回值跟对象、结构、数组所蕴藏的剧情尚未涉及。
    具体来说,当参数分别如下时,sizeof重临的值表示的意义如下:
    数组——编写翻译时分配的数组空间尺寸;
   
指针——存款和储蓄该指针所用的空中尺寸(存款和储蓄该指针的地方的尺寸,是长整型,应该为四);
    类型——该项目所占的半空中尺寸;
    对象——对象的实在占用空间大小;
    函数——函数的回来类型所占的空中山学院小。函数的回到类型不能是void。

sizeof、strlen、字符串、数组,提到那些概念,相信学过C语言的人都能耳熟能详,也能谈得条理鲜明,不过,在其实应用中,当那些剧情交织在联合具名时,大家却不鲜明能搞地清晰,本文的目标就是扶助大家将有关文化总计清楚。


二、strlen
    strlen(…)是函数,要在运作时才具总括。
    参数必须是字符型指针(char*),
且必须是以’\0’结尾的。
当数组名作为参数传入时,实际上数组就退化成指针了。
int ac[10];
    cout<<sizeof(ac)<<endl;
    cout<<strlen(ac)<<endl;    
(ac也等于贰个指南针,可是strlen只好接受char*品种,所以编译时出错)
   
它的效率是:再次来到字符串的长度。该字符串可能是团结定义的,也只怕是内部存款和储蓄器中随机的,该函数实际到位的功效是从代表该字符串的首先个地方初始遍历,直到遭受截止符’\0’。重回的尺寸大小不包涵’\0’。

 

二、strlen
strlen(…)是函数,要在运转时本事总计。参数必须是字符型指针(char*)。当数组名作为参数字传送入时,实际上数组就退化成指针了。
它的职能是:再次回到字符串的长短。该字符串恐怕是团结定义的,也大概是内存中随机的,该函数实际到位的功力是从代表该字符串的率先个地方开首遍历,直到遇见甘休符NULL。再次回到的长短大小不包含NULL。

三、举例:
    eg1、char arr[10] = “Hello”;
              int len_one = strlen(arr);
              int len_two = sizeof(arr); 
              cout << len_one << ” and ” << len_two
<< endl; 
    输出结果为:5 and 拾

正文


    点评:
             sizeof重回定义arr数组时,编写翻译器为其分配的数组空间尺寸,不关切里面存了多少数量。
           
 strlen只关切存储的数据内容,不关注空间的大小和种类。

先看1段代码

三、举例:
eg1、char arr[10] = “What?”;
int len_one = strlen(arr);
int len_two = sizeof(arr);
cout << len_one << ” and ” << len_two <<
endl;
出口结果为:5 and 拾
点评:sizeof重回定义arr数组时,编写翻译器为其分配的数组空间尺寸,不关心里面存了略微多少。strlen只关切存款和储蓄的多少内容,不珍贵空间的分寸和花色。

    eg2、char * parr = new char[10];
              int len_one = strlen(parr);
              int len_two = sizeof(parr);
              int len_three = sizeof(*parr);
              cout << len_one << ” and ” << len_two
<< ” and ” << len_three << endl;
    输出结果:3 and 四 and 壹

 1 #include <stdio.h>
 2 #include <stdlib.h>
 3 
 4 void testchar(char str[])
 5 {
 6     printf("%d %d\n", sizeof(str), strlen(str));
 7 }
 8 
 9 void testint(int arr[])
10 {
11     printf("%d\n", sizeof(arr));
12 }
13 
14 int main()
15 {
16     char str[] = "abc";
17     printf("%d %d\n", sizeof(str), strlen(str)); //4 3
18 
19     char str1[10] = "abc";
20     printf("%d %d\n", sizeof(str1), strlen(str1)); //10 3
21 
22     char dog[] = "wangwang\0miao";
23     printf("%d %d\n", sizeof(dog), strlen(dog)); //14 8
24     testchar(dog); //4 8
25 
26     char *cat = "wangwang\0miaomiao";
27     printf("%d %d\n", sizeof(cat), strlen(cat)); //4 8
28     
29     int arr[10] = { 0 };
30     printf("%d %d\n", sizeof(arr), sizeof(arr[11])); //40 4
31     testint(arr); //4
32 
33     return 0;
34 }
eg2、char * parr = new char[10];
          int len_one = strlen(parr);
          int len_two = sizeof(parr);
          int len_three = sizeof(*parr);
          cout << len_one << " and " << len_two << " and " << len_three << endl;
输出结果:23 and 4 and 1
点评:第一个输出结果23实际上每次运行可能不一样,这取决于parr里面存了什么(从parr[0]开始知道遇到第一个NULL结束);第二个结果实际上本意是想计算parr所指向的动态内存空间的大小,但是事与愿违,sizeof认为parr是个字符指针,因此返回的是该指针所占的空间(指针的存储用的是长整型,所以为4);第三个结果,由于*parr所代表的是parr所指的地址空间存放的字符,所以长度为1。

    点评:
           
 第3个出口结果叁事实上每趟运转恐怕不一致,那有赖于parr里面存了哪些(从parr[0]起来直到遇到第壹个’\0’结束);
           
 第三个结实其实本意是想计算parr所指向的动态内部存款和储蓄器空间的分寸,可是大失所望,sizeof以为parr是个字符指针,因而回到的是该指针所占的空中,指针的贮存用的是长整型,肆字节

 


    
第陆个由于*parr所表示的是parr所指的地点空间存放的字符,所以长度为壹。

 结果

4、参考资料:
Sizeof与Strlen的差别与关系(转)

上面是程序猿面试宝典下边总括的:

 88bifa必发唯一官网 1

壹.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。
该品种保证能包容完毕所创设的最大目的的字节大小。

一.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。 
该品种保障能包容完成所创设的最大目标的字节大小。 

在解说下边包车型大巴事例以前,我们先来说一说sizeof和strlen。

2.sizeof是算符,strlen是函数。

2.sizeof是算符,strlen是函数。 

语法上的真面目不一致:

三.sizeof得以用项目做参数,strlen只好用char*做参数,且必须是以”\0”结尾的。
sizeof还是能用函数做参数,举例:
short f();
printf(“%d\n”, sizeof(f()));
出口的结果是sizeof(short),即二。

三.sizeof方可用项目做参数,strlen只好用char*做参数,且必须是以”\0”结尾的。 
sizeof仍是能够用函数做参数,比如: 
short f(); 
printf(“%d\n”, sizeof(f())); 
出口的结果是sizeof(short),即二。 

sizeof是运算符,strlen是函数。

肆.数组做sizeof的参数不退步,传递给strlen就落5为指针了。

4.数组做sizeof的参数不失利,传递给strlen就落伍为指针了。 

适用范围不均等:

5.多数编写翻译程序 在编译的时候就把sizeof总结过了
是项目或许变量的尺寸那正是sizeof(x)能够用来定义数组维数的缘故
char str[20]=”0123456789″;
int a=strlen(str); //a=10;
int b=sizeof(str); //而b=20;

五.许多编译程序 在编写翻译的时候就把sizeof总结过了
是项目或许变量的尺寸那正是sizeof(x)能够用来定义数组维数的来由 
char str[20]=”0123456789″; 
int a=strlen(str); //a=10; 
int b=sizeof(str); //而b=20; 

对sizeof(name)来说,name能够是变量名也能够是项目名,对strlen来讲,参数必须是char*类别的,即strlen仅用于字符串。

陆.strlen的结果要在运维的时候手艺计算出来,时用来计量字符串的长短,不是项目占内部存款和储蓄器的大大小小。

六.strlen的结果要在运转的时候才干计算出来,时用来计量字符串的尺寸,不是系列占内部存款和储蓄器的大小。 

器重——从底层看本质

7.sizeof后一旦是项目必须加括弧,假如是变量名能够不加括弧。那是因为sizeof是个操作符不是个函数。

柒.sizeof后只倘若类别必须加括弧,纵然是变量名能够不加括弧。这是因为sizeof是个操作符不是个函数。 

strlen(ptr)的实行机理是:从参数ptr所针对的内部存储器开始向下计数,直到内部存款和储蓄器中的内容是全0(即’\0’)为止(不会对’\0’举办计数)。用strlen衡量字符串的长度,其实正是依赖这么些规律。

捌.当适用了于四个布局类型时或变量, sizeof 重回实际的分寸,
当适用一静态地空间数组, sizeof 归还全部数组的尺码。
sizeof 操作符不能回去动态地被分派了的数组或外部的数组的尺寸

8.当适用了于1个组织类型时或变量, sizeof 再次回到实际的轻重, 
当适用壹静态地空间数组, sizeof 归还全部数组的尺寸。 
sizeof 操作符无法回来动态地被分派了的数组或外部的数组的尺码 

sizeof(name)的推行机理是:如若name是二个品种名,得到的是该项目标大大小小(所谓类型的深浅,指的是:如若存在多少个该品种的变量,那个变量在内存中所占用的字节数),假如name是一个变量名,那么,sizeof(name)并不会真的访问该变量,而是先获知该变量的类型,然后再回去该项目标轻重(即就是struct那样的繁杂类型,编译器在编写翻译时也会基于它的各类域记录其尺寸,所以,由项目获得类型大小,不是一件难事)。换句话说,本质上,sizeof的演算对象是项目。倘若name是3个变量名,那么,sizeof如何“对待”name的品种,将是3个关键难题。(前边大家会对那点有深切的认识)

玖.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址,
如:
fun(char [8])
fun(char [])
都相当于 fun(char *)
在C++里参数字传送递数组永久都以传递指向数组首成分的指针,编译器不明白数组的高低
假定想在函数内明白数组的大大小小, 须求如此做:
跻身函数后用memcpy拷贝出来,长度由另3个形参传进去
fun(unsiged char p1, int len)
{
unsigned char
buf = new unsigned char[len+1]
memcpy(buf, p1, len);
}

九.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址, 
如: 
fun(char [8]) 
fun(char []) 
都等于于 fun(char *) 
在C++里参数字传送递数组永世都是传递指向数组首成分的指针,编写翻译器不了然数组的大小 
假定想在函数内精通数组的分寸, 须要这么做: 
进入函数后用memcpy拷贝出来,长度由另二个形参传进去 
fun(unsiged char *p1, int len) 

unsigned char* buf = new unsigned char[len+1] 
memcpy(buf, p1, len); 

地点提到的那或多或少,是知道好sizeof和strlen的不二诀要,是放之所在皆准的规则。上面,大家就以如此的规则来分析上边的例证。

大家能常在用到 sizeof 和 strlen 的时候,平时是持筹握算字符串数组的尺寸
看了地点的详实分解,发掘互相的利用也许有分别的,从那么些例子能够看得很驾驭:

我们能常在用到 sizeof 和 strlen 的时候,平常是估测计算字符串数组的尺寸 
看了地点的事无巨细分解,开采两者的采纳恐怕有分其他,从那么些事例能够看得很领悟: 

a.

char str[20]=”0123456789″;
int a=strlen(str); //a=10; >>>> strlen
计算字符串的尺寸,以截止符 0x00 为字符串结束。
int b=sizeof(str); //而b=20; >>>> sizeof
总结的则是分配的数组 str[20]
所占的内部存储器空间的尺寸,不受里面积攒的故事情节更改。

char str[20]=”0123456789″; 
int a=strlen(str); //a=10; >>>> strlen
总括字符串的长短,以了却符 0x00 为字符串甘休。 
int b=sizeof(str); //而b=20; >>>> sizeof
总计的则是分配的数组 str[20]
所占的内部存款和储蓄器空间的大大小小,不受里面累积的始末改造。 

char str[] = "abc";
printf("%d %d\n", sizeof(str), strlen(str)); //4 3

上边是对静态数组管理的结果,假如是对指针,结果就不等同了

下面是对静态数组管理的结果,假若是对指针,结果就不一样样了 

此地,是用数组的款式评释字符串,编写翻译器会自行在字符串前面加上’\0’,所以,数组的因素个数是4而不是三。对于sizeof(str)来说,sizeof将str视为char
[4]l类型的变量,所以,sizeof(str)的结果就是任何数组所占领的半空中尺寸。对于strlen(str)来讲,它从str指向的内存开头计数,直到遇见全0的内部存款和储蓄器(’\0’),所以最终得到结果3。

char* ss = “0123456789”;
sizeof(ss) 结果 4 ===》ss是指向字符串常量的字符指针,sizeof
得到的是二个指南针的之所占的半空中,应该是

char* ss = “0123456789”; 
sizeof(ss) 结果 肆 ===》ss是指向字符串常量的字符指针,sizeof
拿到的是贰个指南针的之所占的空中,应该是 长整型的,所以是四 
sizeof(*ss) 结果 1 ===》*ss是首先个字符
其实便是获得了字符串的首先位’0′ 所占的内部存储器空间,是char类 型的,占了 1位 
strlen(ss)= 十 >>>> 借使要赚取那么些字符串的长度,则断定要利用
strlen。

b.

长整型的,所以是4
sizeof(ss) 结果 1 ===》ss是第贰个字符
其实正是获得了字符串的首先位’0′ 所占的内部存款和储蓄器空间,是char类

 

char str1[10] = "abc";
printf("%d %d\n", sizeof(str1), strlen(str1)); //10 3

型的,占了 1 位

转发外人链接:

88bifa必发唯一官网, 

strlen(ss)= 10 >>>> 借使要赢得那一个字符串的尺寸,则必定要选取strlen

编译器为char
str一[10]分红十二个数组成分大小的长空,那与开端化它的字符串未有提到,所以sizeof(str1)得到10。

转自http://www.cnblogs.com/carekee/articles/1630789.html

c.

char dog[] = "wangwang\0miao";
printf("%d %d\n", sizeof(dog), strlen(dog)); //14 8
testchar(dog); //4 8

前两句和a中的情形亦然,sizeof(dog)输出整个数组所占的内部存款和储蓄器大小(包罗编写翻译器加上去的’\0’),strlen(dog)遇到’\0’就止住,所以输出八。

再看前面包车型大巴函数调用,数组名dog作为函数实参传入,大家再来回想一下testchar函数

void testchar(char str[])
{
    printf("%d %d\n", sizeof(str), strlen(str));
}

咱俩发掘,这里sizeof(str)并未像sizeof(dog)那样获得1肆,而是得到了四。那是因为,str是函数形参,尽管它是以数组名的款式出现的,传给它的实参也实在是数组名,但sizeof仅仅把它当成3个char*品类的指针对待,所以,sizeof(str)的结果正是char
*花色所占的空间四。至于strlen(str),大家前面说过,它实施的机理正是从str指向的内部存款和储蓄器早先向下计数,直到遇见’\0’,所以依然收获八。

d.

char *cat = "wangwang\0miaomiao";
printf("%d %d\n", sizeof(cat), strlen(cat)); //4 8 

是因为cat明确宣示为char*,所以sizeof将它视为指针,获得肆。

e.

int arr[10] = { 0 };
printf("%d %d\n", sizeof(arr), sizeof(arr[11])); //40 4
testint(arr); //4

前方说过,当数组名作为函数形参出现时,sizeof仅仅将其正是贰个指南针,不然,sizeof感觉它表示任何数组,所以,sizeof(arr)获得全数数组所占的字节数40,而testint(arr)的结果是int*类型的指针的长度四。

sizeof(int[11])中,很显明数组越界了,但并不会油不过生运转时不当。原因是:依附大家提交的论断准则,sizeof并从未真的访问arr[11],根据arr的声明,sizeof知道arr[11]是int型的,所以回来int类型的轻重缓急。

至于testint(arr),道理和c中的testchar(dog)相同。

 

末段,基于上边的座谈,给出编码准则:

1.千古不要用sizeof来求字符串长度!它不是干那些活的,所以您也永久不会获取不错答案。

二.不要自作聪明地用sizeof(arr)/sizeof(arr[0])那样的代码求数组的长短!sizeof也不是干这一个活的。如果arr是函数形参,获得的结果将是谬误的(除非你在三十三人系统下恰好注解int
arr[1]或者char
arr[4]等,但这纯属巧合)。既然是数组,长度自然是已知的,求数老总度那1本身,就是神经过敏的工巧行为。

 

写在背后

本文的目的,便是使读者对C语言的基础知识——sizeof和strlen有一个真相的认知,同时对与之城门失火的易错、易混难题有三个科学、清晰的判断。由于在下才疏学浅,错误疏漏之处在所难免,希望广大读者积极争辩指正,您的批评指便是在下发展的不竭重力。