一、字符数组初始化与sizeof、strlen
- 字符数组就是字符类型的数组
- sizeof是c语言的一个关键字和运算符。sizeof(类型、变量名),很多人误以为它是函数。
- sizeof运算符用来返回一个类型或者变量所占的内存字节数。int double等原生类型占几个字节和平台有关;c语言中除了ADT还有UDT,这些用户自定义类型占几个字节无法一眼看出,所以用sizeof运算符来让编译器帮忙计算。
- strlen是c语言的一个库函数,这个库函数的原型是:size_t strlen(const char *s);这个函数接收一个字符串的指针,返回字符串的长度(以字节为单位)返回字符串长度是不包括字符串结尾。
- 字符串定义(指针指向头、固定结尾、中间依次相连)可知,无法直接得到字符串长度,需要用strlen来计算得到字符串长度。
- sizeof(数组名),得到的永远是数组的个数(也就是数组的大小),和数组中有无初始化,初始化多还是少是没有关系的。
- strlen只能用来计算字符串的长度,只能传递合法的字符串指针进去才有意义,用strlen测量字符数组的长度是不合适的。
- 当我们定义数组时如果没有明确给出数组大小,则必须同时给出初始化式,编译器会根据初始化式去自动计算数组大小。
#include <stdio.h>
#include <string.h>
int mystrlen(const char *p)
{
int cnt=0;
while(*p++!='\0')
{
cnt++;
//p++;
}
return cnt;
}
int main(void)
{
char *p="linux";
//int len=strlen(p);
int len=mystrlen(p);
printf("len=%d.\n",len); //5
return 0;
}
#include <stdio.h>
#include <string.h>
int main(void)
{
char a[5]; //定义字符数组
char a[5]={0};
char a[5]={2,3};
char a[5]="lin";
char a[5]="windows";
char a[]="windows";
char a[7]="windows";
printf("sizeof(a)=%d.\n",sizeof(a)); //5,5,5,5,5,8,7
printf("strlen(a)=%d.\n",strlen(a)); //5,0,2,3,5,7,7
return 0;
}
二、字符串初始化与sizeof、strlen
- 字符串是没法用sizeof运算符测量长度的,因为p是一个指针测出来永远是4和字符串长度无关。字符串长度非要借助strlen函数测量。
#include <stdio.h>
int main(void)
{
char *p="linux";
char *p="linuxddd";
printf("sizeof(p)=%d.\n",sizeof(p)); //4,4 p本身是一个指针
printf("strlen(p)=%d.\n",strlen(p)); //5,8
return 0;
}
三、字符数组与字符串的本质差异(内存分配角度)
- 字符串和字符数组在很多地方上几乎是相同的,不同点主要体现在内存角度上。
- 字符数组char a[]="linux"; 定义了一个数组a,a占6个字节,右值"linux"本身只存在编译器中,编译器把它用于初始化数组后丢弃(内存里没有”linux"这个字符串)。相当于a[]={'l','i','n','u','x'};
- 字符串char *p="linux";定义了一个字符指针p,p占4个字节,分配在栈上,同时还定义了一个字符串“linux”,分配在代码段;然后把代码段的字符串(一共占了6个字节)首地址赋值给p。
- 字符数组和字符串有本质差别。字符数组自带内存空间,可以用来存东西。而字符串本身是指针,本身永远只占4个字节,不能用来存有效数据,只能把有效数据存在别的地方,然后把地址存在p中。
- 字符数组自己存字符,字符串需要额外内存存储字符。字符串本身只存字符所在内存的首地址。
- 字符串可以定义到不同的内存中,二字符数字只能存储在栈上。
#include <stdio.h>
#include <string.h>
char b[5];
int main(void)
{
//把字符串存储到栈上
char a[7];
char *p=a;
//把字符串存储到栈上
char *p=b;
//把字符串存储到堆上
char *p=(char *)malloc(5);
return 0;
}