sizeof、strlen、字符串、数组,整到一块,你还清楚吗?

写在前面

sizeof、strlen、字符串、数组,提到这些概念,相信学过C语言的人都能耳熟能详,也能谈得头头是道,但是,在实际运用中,当这些内容交织在一起时,大家却不一定能搞地清清楚楚,本文的目的正是帮助大家将相关知识总结清楚。

正文

先看一段代码

 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 }

结果

在解释上面的例子之前,我们先来说一说sizeof和strlen。

语法上的本质不同:

sizeof是运算符,strlen是函数。

适用范围不一样:

对sizeof(name)而言,name可以是变量名也可以是类型名,对strlen而言,参数必须是char*类型的,即strlen仅用于字符串。

重中之重——从底层看本质

strlen(ptr)的执行机理是:从参数ptr所指向的内存开始向下计数,直到内存中的内容是全0(即’\0’)为止(不会对’\0’进行计数)。用strlen测量字符串的长度,其实就是基于这个原理。

sizeof(name)的执行机理是:如果name是一个类型名,得到的是该类型的大小(所谓类型的大小,指的是:如果存在一个该类型的变量,这个变量在内存中所占用的字节数),如果name是一个变量名,那么,sizeof(name)并不会真正访问该变量,而是先获知该变量的类型,然后再返回该类型的大小(即便是struct这样的复杂类型,编译器在编译时也会根据它的各个域记录其大小,所以,由类型得到类型大小,不是一件难事)。换句话说,本质上,sizeof的运算对象是类型。如果name是一个变量名,那么,sizeof如何“看待”name的类型,将是一个关键问题。(后面我们会对这一点有深刻的体会)

上面提到的这一点,是理解好sizeof和strlen的不二法门,是放之四海皆准的准则。下面,我们就以这样的准则来分析上面的例子。

a.

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

这里,是用数组的形式声明字符串,编译器会自动在字符串后面加上‘\0‘,所以,数组的元素个数是4而不是3。对于sizeof(str)而言,sizeof将str视为char [4]l类型的变量,所以,sizeof(str)的结果就是整个数组所占有的空间大小。对于strlen(str)来说,它从str指向的内存开始计数,直到遇到全0的内存(‘\0‘),所以最后得到结果3。

b.

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

编译器为char str1[10]分配10个数组元素大小的空间,这与初始化它的字符串没有关系,所以sizeof(str1)得到10。

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‘就停止,所以输出8。

再看后面的函数调用,数组名dog作为函数实参传入,我们再来回顾一下testchar函数

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

我们发现,这里sizeof(str)并没有像sizeof(dog)那样得到14,而是得到了4。这是因为,str是函数形参,尽管它是以数组名的形式出现的,传给它的实参也确实是数组名,但sizeof仅仅把它当成一个char*类型的指针看待,所以,sizeof(str)的结果就是char *类型所占的空间4。至于strlen(str),我们前面说过,它执行的机理就是从str指向的内存开始向下计数,直到遇到‘\0‘,所以依然得到8。

d.

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

由于cat明确声明为char*,所以sizeof将它视为指针,得到4。

d.

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),道理和b中的testchar(dog)相同。

最后,基于上面的讨论,给出编码准则:

1.永远不要用sizeof来求字符串长度!它不是干这个活的,所以你也永远不会得到正确答案。

2.不要自作聪明地用sizeof(arr)/sizeof(arr[0])这样的代码求数组的长度!sizeof也不是干这个活的。如果arr是函数形参,得到的结果将是错误的(除非你在32位系统下恰好声明int arr[1]或者char arr[4]等,但这纯属巧合)。既然是数组,长度自然是已知的,求数组长度这一本身,就是独此一举的愚蠢行为。

写在后面

本文的目的,就是使读者对C语言的基础知识——sizeof和strlen有一个本质的认识,同时对与之相关的易错、易混问题有一个正确、清晰的判断。由于在下才疏学浅,错误疏漏之处在所难免,希望广大读者积极批评指正,您的批评指正是在下前进的不竭动力。

时间: 2024-08-08 05:17:48

sizeof、strlen、字符串、数组,整到一块,你还清楚吗?的相关文章

使用 sizeof 获取字符串数组的大小

@2018-11-1 字符串组成的数组存放于指针数组中,使用 sizeof 获取数组大小 [验证] 1 #include <stdio.h> 2 3 4 #define BootScreen "1:开机画面.jpg" 5 #define Computer0 "1:computer0.dta" 6 #define Linkage0 "1:linkage0.dta" 7 #define Linkage1 "1:linkage1.

PHP的count(数组)和strlen(字符串)的内部实现

PHP的count(数组)和strlen(字符串)的内部实现上是直接显示一个长度变量,还是重头依次数一遍有多少个元素?关乎我理解这2个函数的效率..希望高人能从php的c源码上讲一讲.没有源码看过源码知道的说说也行. 1.count执行时背后有没有“逐个统计”子元素的个数?2.strlen执行时背后有没有“逐个统计”字符的个数?==========我已经找到PHP strlen的源码了,唉沉下去的没几人...正在找count的=================结论strlen直接返回长度变量,没

strlen() 和 sizeof() 在字符串中的使用

#include <string.h> int _tmain(int argc, _TCHAR* argv[]) { char *pMyChar = "I like coding !"; char myChar[] = "I like coding !"; int num1(0),num2(0); num1 = sizeof(pMyChar); //num1=4,32位系统下指针占4个字节 num2 = sizeof(myChar); //num2=16

求字符串长度 strlen(数组指针两种方式)

问题: 求字符串中所含有字符的个数(包括空格),即求字符串长度: #include <stdio.h> #include <assert.h> int _strlen(const char* str) { assert(str != NULL); int i=0; for(;*str++!='\0';i++); //for(;str++!=NULL;i++);//有些说这句也可以,但执行结果是死循环,str++即使越界也未必为NULL; return i; } int _strle

C语言数组:C语言数组定义、二维数组、动态数组、字符串数组

1.C语言数组的概念 在<更加优美的C语言输出>一节中我们举了一个例子,是输出一个 4×4 的整数矩阵,代码如下: #include <stdio.h> #include <stdlib.h> int main() { int a1=20, a2=345, a3=700, a4=22; int b1=56720, b2=9999, b3=20098, b4=2; int c1=233, c2=205, c3=1, c4=6666; int d1=34, d2=0, d3

sizeof/strlen/length

1.sizeof() sizeof(...)是运算符,在头文件中typedef为unsigned int,其值在编译时即计算好了,参数可以是数组.指针.类型.对象.函数等. 它的功能是:获得保证能容纳实现所建立的最大对象的字节大小. 由于在编译时计算,因此sizeof不能用来返回动态分配的内存空间的大小.实际上,用sizeof来返回类型以及静态分配的对象.结构或数组所占的空间,返回值跟对象.结构.数组所存储的内容没有关系. 具体而言,当参数分别如下时,sizeof返回的值表示的含义如下: 数组—

sizeof, strlen区别

strlen与sizeof的区别 1.sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned int类型. 该类型保证能容纳实现所建立的最大对象的字节大小. 2.sizeof是算符,strlen是函数. 3.sizeof可以用类型做参数,strlen只能用char*做参数,且必须是以''\0''结尾的. 4.数组做sizeof的参数不退化,传递给strlen就退化为指针了. 5.大部分编译程序在编译的时候就把sizeof计算过了 是类型或是变量的长度这就是siz

字符串数组初始化相关问题总结

C字符串数组初始化相关问题总结 在C语言编程中,当我们声明一个字符串数组的时候,常常需要把它初始化为空串.总结起来有以下三种方式: (1) char str[10]=""; (2) char str[10]={'\0'}; (3) char str[10]; str[0]='\0'; 第(1)(2)种方式是将str数组的所有元素都初始化为'\0',而第(3)种方式是只将str数组的第一个元素初始化为'\0'.如果数组的size非常大,那么前两种方式将会造成很大的开销. 所以,除非必要(

iOS开发入门 ? C语言(字符串、字符串数组、命令行参数)

字符串 1. 概念 用双引号引起来的就是字符串 "a string" //C语言编译器会将两个并列的字符串自动拼接成一个字符串 "a string""another a string" // \是换行连接的标记(一行不够写) "a looooooooooong \ string" 常见ASCII编码: 'A' == 65    'a' == 97    '0' == 48    '\0' == 0 int a[10]; //表

八、字符串数组

1.用于存放字符变量的数组称为字符数组 2.字符数组属于一维数组,类型char 3.字符串数组:是存放字符串的数组 4.字符串数组定义: char 数组名[常量表达式1][常量表达式2] = {值1,值2,...}: chat strings[3][10] = {"iphone","Android","win8"};包含三个字符串 5.字符串数组本质就是字符型二维数组.一个包含多个字符数组的数组 6.字符数组,行代表的是每个字符串,列代表的是字符