Sizeof与Strlen的区分与联系,整到一块

写在前边

一、sizeof
sizeof(…)是运算符,在头文件中typedef为unsigned
int,其值在编写翻译时即总计好了,参数能够是数组、指针、类型、对象、函数等。
它的效益是:获得保证能包容落成所创造的最大指标的字节大小。
鉴于在编写翻译时总结,由此sizeof不能够用来回到动态分配的内存空间的轻重。实际上,用sizeof来回到类型以及静态分配的目标、结构或数组所占的上空,重临值跟对象、结构、数组所蕴藏的内容尚未涉嫌。
具体而言,当参数分别如下时,sizeof重返的值表示的意思如下:
数组——编写翻译时分配的数组空间大小;
指南针——存款和储蓄该指针所用的长空尺寸(存款和储蓄该指针的地点的长短,是长整型,应该为四);
类型——该品种所占的上台湾空中大学小;
目的——对象的莫过于占用空间尺寸;
函数——函数的回到类型所占的半空中山大学小。函数的归来类型不可能是void。

一、sizeof
    sizeof(…)是运算符,而不是三个函数。
    贰个简便的例证:
int a;
cout<<sizeof a<<endl;
    在头文件中typedef为unsigned
int,其值在编写翻译时即计算好了,参数能够是数组、指针、类型、对象、函数等。
    它的功力是:收获有限支撑能包容完毕所树立的最大目的的字节大小
 
  鉴于在编写翻译时总括,因而sizeof无法用来回到动态分配的内部存款和储蓄器空间的轻重缓急。
   
实际上,用sizeof来回到类型以及静态分配的靶子、结构或数组所占的长空,重临值跟对象、结构、数组所蕴藏的剧情未有提到。
    具体而言,当参数分别如下时,sizeof重返的值表示的含义如下:
    数组——编写翻译时分配的数组空间大小;
   
指针——存储该指针所用的空中山大学小(存储该指针的地方的尺寸,是长整型,应该为四);
    类型——该项目所占的半空中山大学小;
    对象——对象的骨子里占有空间尺寸;
    函数——函数的回来类型所占的空中山大学小。函数的回到类型不能够是void。

一、sizeof
    sizeof(…)是运算符,在头文件中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。
*****************

 

二、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; 
    输出结果为:伍 and 拾

三、举例:
    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与Strlen的区分与联系,整到一块。    点评:sizeof重返定义arr数组时,编写翻译器为其分配的数组空间尺寸,不关怀里面存了稍稍数量。strlen只关怀存款和储蓄的数码内容,不关心空间的分寸和项目。

正文


    点评:
             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;
    输出结果:23 and 四 and 一
    点评:第2个出口结果贰3实际每一趟运转大概不均等,那取决parr里面存了哪些(从parr[0]开端精晓遇到第一个NULL截止);第三个结果其实本意是想总计parr所指向的动态内部存款和储蓄器空间的大大小小,不过差强人意,sizeof认为parr是个字符指针,因而回到的是该指针所占的上空(指针的储存用的是长整型,所以为四);第四个结果,由于*parr所表示的是parr所指的地点空间存放的字符,所以长度为一。
************

先看壹段代码

三、举例:
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 壹

4、参考资料:
Sizeof与Strlen的区分与联系(转) 

 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是个字符指针,由此回到的是该指针所占的空中,指针的囤积用的是长整型,四字节

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

 


    
第三个由于*parr所代表的是parr所指的地址空间存放的字符,所以长度为一。

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

 结果

四、参考资料:
Sizeof与Strlen的分别与交流(转)

下边是程序员面试宝典上边计算的:

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

 美高梅开户网址 1

一.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。
该项目保险能包容完结所树立的最大目标的字节大小。

1.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。 
该类型保障能包容完毕所树立的最大目的的字节大小。 

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

在诠释下边的例证在此之前,大家先来说1说sizeof和strlen。

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

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

伍.当先贰分一编写翻译程序 在编写翻译的时候就把sizeof总括过了 是项目也许变量的长短这正是sizeof(x)能够用来定义数组维数的原由 
char str[20]=”0123456789″; 
int a=strlen(str); //a=10; 
int b=sizeof(str); //而b=20; 

语法上的昆仑山真面目区别:

叁.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),即二。 

陆.strlen的结果要在运作的时候才能总括出来,时用来总结字符串的长度,不是项目占内部存款和储蓄器的分寸。 

sizeof是运算符,strlen是函数。

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

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

七.sizeof后假若是连串必须加括弧,假设是变量名能够不加括弧。那是因为sizeof是个操作符不是个函数。 

适用范围不一样:

五.一大半编写翻译程序 在编写翻译的时候就把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; 

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

对sizeof(name)而言,name能够是变量名也得以是项目名,对strlen而言,参数必须是char*品类的,即strlen仅用于字符串。

陆.strlen的结果要在运作的时候才能总结出来,时用来测算字符串的长度,不是类别占内部存款和储蓄器的深浅。

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

九.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址, 
如: 
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); 

最重要——从底部看本质

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

柒.sizeof后一旦是种类必须加括弧,假如是变量名可以不加括弧。这是因为sizeof是个操作符不是个函数。 

小编们能常在用到 sizeof 和 strlen 的时候,经常是测算字符串数组的长短 
看了地方的事无巨细表达,发现两者的施用大概有分别的,从这些例子可以看得很精晓: 

strlen(ptr)的实践机理是:从参数ptr所指向的内部存款和储蓄器开头向下计数,直到内部存款和储蓄器中的内容是全0(即’\0’)为止(不会对’\0’进行计数)。用strlen度量字符串的长短,其实就是基于这些规律。

八.当适用了于2个协会类型时或变量, sizeof 重回实际的轻重,
当适用一静态地空间数组, sizeof 归还全体数组的尺寸。
sizeof 操作符不能回来动态地被分派了的数组或外部的数组的尺码

八.当适用了于三个构造类型时或变量, sizeof 重临实际的轻重缓急, 
当适用一静态地空间数组, sizeof 归还全部数组的尺寸。 
sizeof 操作符无法重返动态地被分派了的数组或外部的数组的尺寸 

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

sizeof(name)的推行机理是:若是name是一个项目名,获得的是该品种的轻重缓急(所谓类型的尺寸,指的是:倘使存在二个该类型的变量,这一个变量在内部存款和储蓄器中所占用的字节数),要是name是1个变量名,那么,sizeof(name)并不会真的访问该变量,而是先获知该变量的种类,然后再回来该项指标尺寸(即正是struct那样的繁杂类型,编写翻译器在编写翻译时也会依照它的种种域记录其尺寸,所以,由项目获得类型大小,不是1件难事)。换句话说,本质上,sizeof的运算对象是项目。要是name是三个变量名,那么,sizeof如何“看待”name的类型,将是叁个关键难点。(前边大家会对那点有深入的认知)

玖.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址,
如:
fun(char [8])
fun(char [])
都约等于 fun(char *)
在C++里参数字传送递数组永远都是传递指向数组首成分的指针,编译器不驾驭数组的分寸
万1想在函数内驾驭数组的轻重, 需求这么做:
进入函数后用memcpy拷贝出来,长度由另3个形参传进去
fun(unsiged char p1, int len)
{
unsigned char
buf = new unsigned char[len+1]
memcpy(buf, p1, len);
}

9.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址, 
如: 
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 的时候,经常是一个钱打二十七个结字符串数组的长短 
看了地方的详实表达,发现相互的利用也许有分其他,从这一个例子能够看得很清楚: 

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

a.

char str[20]=”0123456789″;
int a=strlen(str); //a=十; >>>> strlen
总结字符串的长短,以甘休符 0x00 为字符串甘休。
int b=sizeof(str); //而b=20; >>>> sizeof
计算的则是分配的数组 str[20]
所占的内部存款和储蓄器空间的轻重缓急,不受里面储存的剧情变更。

char str[20]=”0123456789″; 
int a=strlen(str); //a=十; >>>> strlen
总计字符串的长度,以了却符 0x00 为字符串结束。 
int b=sizeof(str); //而b=20; >>>> sizeof
总结的则是分配的数组 str[20]
所占的内部存款和储蓄器空间的深浅,不受里面储存的始末改动。 

长整型的,所以是四 
sizeof(*ss) 结果 1 ===》*ss是首先个字符 其实正是获取了字符串的率先位’0′ 所占的内存空间,是char类 

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

地点是对静态数组处理的结果,倘使是对指针,结果就分化等了

地点是对静态数组处理的结果,假诺是对指针,结果就不雷同了 

型的,占了 1 位 

此处,是用数组的样式申明字符串,编写翻译器会活动在字符串后面加上’\0’,所以,数组的成分个数是4而不是叁。对于sizeof(str)而言,sizeof将str视为char
[4]l类型的变量,所以,sizeof(str)的结果正是全方位数组所占用的空中尺寸。对于strlen(str)来说,它从str指向的内部存款和储蓄器早先计数,直到蒙受全0的内部存款和储蓄器(’\0’),所以最终收获结果三。

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

char* ss = “0123456789”; 
sizeof(ss) 结果 四 ===》ss是指向字符串常量的字符指针,sizeof
得到的是3个指南针的之所占的空间,应该是 长整型的,所以是4 
sizeof(*ss) 结果 1 ===》*ss是率先个字符
其实就是获取了字符串的第二位’0′ 所占的内部存款和储蓄器空间,是char类 型的,占了 1人 
strlen(ss)= 10 >>>> 借使要收获这一个字符串的长短,则终将要采纳strlen。

strlen(ss)= 10>>>> 假若要博得这一个字符串的尺寸,则一定要动用 strlen

b.

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

 

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

型的,占了 1 位

转载外人链接:

 

strlen(ss)= 10 >>>> 假若要博取这些字符串的尺寸,则必定要动用
strlen

编译器为char
str1[10]分红13个数组成分大小的半空中,这与开始化它的字符串未有涉及,所以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肆,而是得到了4。那是因为,str是函数形参,就算它是以数组名的格局现身的,传给它的实参也确实是数组名,但sizeof仅仅把它就是2个char*花色的指针看待,所以,sizeof(str)的结果便是char
*项目所占的空间4。至于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*类别的指针的长度4。

sizeof(int[11])中,很扎眼数组越界了,但并不会油但是生运维时不当。原因是:依据大家付出的判断准则,sizeof并未当真访问arr[11],根据arr的声明,sizeof知道arr[11]是int型的,所以回来int类型的轻重。

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

 

末段,基于上边的议论,给出编码准则:

一.永远不要用sizeof来求字符串长度!它不是干那几个活的,所以你也永远不会博得不错答案。

二.不要布鼓雷门地用sizeof(arr)/sizeof(arr[0])那样的代码求数组的长短!sizeof也不是干那个活的。若是arr是函数形参,得到的结果将是荒谬的(除非你在3十人系统下恰好注解int
arr[1]或者char
arr[4]等,但那纯属巧合)。既然是数组,长度自然是已知的,求数老董度这一本身,就是舍近求远的鲁钝行为。

 

写在末端

本文的目标,正是使读者对C语言的基础知识——sizeof和strlen有四个实质的认识,同时对与之有关的易错、易混难点有3个不错、清晰的判断。由于在下才疏学浅,错误疏漏之处在所难免,希望广大读者积极批评指正,您的批评指正是在下发展的不竭引力。

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图