写在前边
一、sizeof
sizeof(…)是运算符,在头文件中typedef为unsigned
int,其值在编写翻译时即总结好了,参数能够是数组、指针、类型、对象、函数等。
它的职能是:得到保险能包容实现所创设的最大指标的字节大小。
Sizeof与Strlen的区分与联系,你还知道啊。出于在编写翻译时总括,因而sizeof无法用来回到动态分配的内部存款和储蓄器空间的轻重。实际上,用sizeof来回到类型以及静态分配的目的、结构或数组所占的空间,重临值跟对象、结构、数组所蕴藏的内容尚未关系。
具体而言,当参数分别如下时,sizeof再次回到的值表示的意思如下:
数组——编译时分配的数组空间大小;
指南针——存款和储蓄该指针所用的上台湾空中大学小(存款和储蓄该指针的地点的长短,是长整型,应该为四);
类型——该类型所占的上台湾空中大学小;
指标——对象的莫过于占有空间尺寸;
函数——函数的回到类型所占的长台湾空中大学小。函数的归来类型无法是void。
一、sizeof
sizeof(…)是运算符,而不是几个函数。
2个简练的例证:
int a;
cout<<sizeof a<<endl;
在头文件中typedef为unsigned
int,其值在编写翻译时即计算好了,参数能够是数组、指针、类型、对象、函数等。
它的机能是:收获保障能包容实现所树立的最大指标的字节大小。
鉴于在编写翻译时总计,由此sizeof不能够用来回到动态分配的内部存储器空间的轻重缓急。
实际上,用sizeof来回到类型以及静态分配的对象、结构或数组所占的上空,再次来到值跟对象、结构、数组所蕴藏的始末从未提到。
具体而言,当参数分别如下时,sizeof重返的值表示的意义如下:
数组——编写翻译时分配的数组空间大小;
指针——存款和储蓄该指针所用的半空中山大学小(存款和储蓄该指针的地点的尺寸,是长整型,应该为4);
类型——该项目所占的长台湾空中大学小;
对象——对象的骨子里占有空间尺寸;
函数——函数的回来类型所占的半空中山大学小。函数的回到类型不可能是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 10
三、举例:
eg1、char arr[10] = “What?”;
int len_one = strlen(arr);
int len_two = sizeof(arr);
cout << len_one << ” and ” << len_two
<< endl;
输出结果为:伍 and 拾
点评: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;
输出结果:贰三 and 4 and 一
点评:第二个出口结果贰三其实每回运营也许不均等,那取决于parr里面存了什么样(从parr[0]起头通晓遇到第一个NULL甘休);第一个结实其实本意是想总括parr所指向的动态内部存款和储蓄器空间的轻重缓急,不过壮志未酬,sizeof认为parr是个字符指针,由此回到的是该指针所占的空中(指针的贮存用的是长整型,所以为四);第二个结果,由于*parr所代表的是parr所指的地点空间存放的字符,所以长度为一。
************
先看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;
输出结果:三 and 四 and 一
四、参考资料:
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是个字符指针,因而回到的是该指针所占的长空,指针的囤积用的是长整型,4字节
一.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。
该类型有限支撑能包容完成所确立的最大指标的字节大小。
第一个由于*parr所表示的是parr所指的地方空间存放的字符,所以长度为一。
2.sizeof是算符,strlen是函数。
结果
四、参考资料:
Sizeof与Strlen的区分与关系(转)
下边是程序员面试宝典上边计算的:
三.sizeof得以用项目做参数,strlen只好用char*做参数,且必须是以”\0”结尾的。
sizeof还能用函数做参数,比如:
short f();
printf(“%d\n”, sizeof(f()));
出口的结果是sizeof(short),即2。
壹.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。
该类型保险能包容达成所成立的最大目的的字节大小。
1.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned
int类型。
该品种保险能包容完结所创造的最大目的的字节大小。
四.数组做sizeof的参数不失利,传递给strlen就落五为指针了。
在解说下面的例子此前,大家先来说壹说sizeof和strlen。
2.sizeof是算符,strlen是函数。
2.sizeof是算符,strlen是函数。
5.多数编写翻译程序 在编译的时候就把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),即2。
三.sizeof得以用项目做参数,strlen只可以用char*做参数,且务必是以”\0”结尾的。
sizeof还足以用函数做参数,比如:
short f();
printf(“%d\n”, sizeof(f()));
输出的结果是sizeof(short),即贰。
6.strlen的结果要在运营的时候才能总计出来,时用来总结字符串的尺寸,不是项目占内部存款和储蓄器的轻重缓急。
sizeof是运算符,strlen是函数。
四.数组做sizeof的参数不落5,传递给strlen就落5为指针了。
四.数组做sizeof的参数不落5,传递给strlen就落后为指针了。
柒.sizeof后1旦是类别必须加括弧,如若是变量名能够不加括弧。那是因为sizeof是个操作符不是个函数。
适用范围不平等:
5.多数编写翻译程序 在编写翻译的时候就把sizeof总结过了
是体系可能变量的尺寸这正是sizeof(x)能够用来定义数组维数的缘由
char str[20]=”0123456789″;
int a=strlen(str); //a=10;
int b=sizeof(str); //而b=20;
5.超越二分一编写翻译程序 在编写翻译的时候就把sizeof总括过了
是连串或然变量的尺寸那就是sizeof(x)可以用来定义数组维数的原因
char str[20]=”0123456789″;
int a=strlen(str); //a=10;
int b=sizeof(str); //而b=20;
8.当适用了于3个构造类型时或变量, 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);
}
要害——从底层看本质
七.sizeof后只借使项目必须加括弧,假如是变量名能够不加括弧。这是因为sizeof是个操作符不是个函数。
七.sizeof后假如是种类必须加括弧,如若是变量名能够不加括弧。那是因为sizeof是个操作符不是个函数。
大家能常在用到 sizeof 和 strlen 的时候,平日是计量字符串数组的长短
看了上面的详细解释,发现三头的应用如故有分其余,从那一个事例能够看得很明亮:
strlen(ptr)的实施机理是:从参数ptr所指向的内部存款和储蓄器起头向下计数,直到内部存款和储蓄器中的内容是全0(即’\0’)为止(不会对’\0’举办计数)。用strlen度量字符串的长短,其实正是依照这几个原理。
八.当适用了于二个结构类型时或变量, sizeof 重返实际的尺寸,
当适用一静态地空间数组, sizeof 归还全体数组的尺码。
sizeof 操作符不能回到动态地被分派了的数组或外部的数组的尺码
8.当适用了于一个协会类型时或变量, sizeof 重临实际的轻重,
当适用1静态地空间数组, 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是3个变量名,那么,sizeof(name)并不会真的访问该变量,而是先获知该变量的花色,然后再回去该品种的大大小小(即就是struct那样的复杂性类型,编写翻译器在编写翻译时也会基于它的各样域记录其尺寸,所以,由项目获得类型大小,不是1件难事)。换句话说,本质上,sizeof的运算对象是类别。假诺name是三个变量名,那么,sizeof怎么样“看待”name的连串,将是3个关键难题。(前面大家会对那一点有深刻的体味)
玖.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址,
如:
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);
}
玖.数组作为参数字传送给函数时传的是指针而不是数组,传递的是数组的首地址,
如:
fun(char [8])
fun(char [])
都约等于 fun(char *)
在C++里参数字传送递数组永远都以传递指向数组首成分的指针,编写翻译器不知道数组的大小
假定想在函数内驾驭数组的尺寸, 必要如此做:
进去函数后用memcpy拷贝出来,长度由另1个形参传进去
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) 结果 4 ===》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是率先个字符 其实便是获得了字符串的第2个人’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) 结果 四 ===》ss是指向字符串常量的字符指针,sizeof
获得的是2个指针的之所占的长空,应该是
char* ss = “0123456789”;
sizeof(ss) 结果 四 ===》ss是指向字符串常量的字符指针,sizeof
获得的是多少个指针的之所占的半空中,应该是 长整型的,所以是四
sizeof(*ss) 结果 1 ===》*ss是第多个字符
其实正是赢得了字符串的首先位’0′ 所占的内部存款和储蓄器空间,是char类 型的,占了 一位
strlen(ss)= 拾 >>>> 假若要获取这一个字符串的长度,则早晚要运用
strlen。
strlen(ss)= 10>>>> 借使要拿到那一个字符串的长短,则肯定要选择 strlen
b.
长整型的,所以是肆
sizeof(ss) 结果 1 ===》ss是率先个字符
其实就是获得了字符串的第1人’0′ 所占的内部存款和储蓄器空间,是char类
char str1[10] = "abc";
printf("%d %d\n", sizeof(str1), strlen(str1)); //10 3
型的,占了 1 位
转发别人链接:
strlen(ss)= 10 >>>> 假若要获取那个字符串的长短,则必定要运用
strlen
编写翻译器为char
str一[10]分配13个数组成分大小的上空,那与开头化它的字符串未有关系,所以sizeof(str一)获得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仅仅把它便是一个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仅仅将其视为2个指南针,不然,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]等,但那纯属巧合)。既然是数组,长度自然是已知的,求数经理度那壹本人,正是大做文章的愚笨行为。
写在末端
本文的指标,就是使读者对C语言的基础知识——sizeof和strlen有三个实质的认识,同时对与之有关的易错、易混难点有叁个毋庸置疑、清晰的论断。由于在下才疏学浅,错误疏漏之处在所难免,希望广大读者积极批评指正,您的批评指就是在下发展的不竭引力。