仔细的看了指针和数组的关系,也知道字符串也是一个数组,是字符拼凑的数组,所以也把字符串也叫字符数组。
先来复习下如何声明一个字符串?
char s[] = {'m','j', '\0'};
char s[] = "mj"; //另一种方式。自动在最后加\0
在内存的中的展现形式就是:
由于,字符串其实也是一个数组,数组与指针的亲密关系,那与字符串关系理论上也是一样。
指针指向字符串
看下,如何用一个指针指向一个字符串,其实和数组一样:
char s[] = "mj"; //定义一个字符串
char *p = s; //定义一个指针p指向 字符串 s
你看,是不是和数组的一模一样。在内存中的呈现,也是一样的:
所以,p++ 也是移动指针,指向字符串中的下一个元素,由于字符占一个字节的空间,所以,p+1也就只需要移动一个空间的位置。*(p+1)就是取下一个元素的值。
来一个例子,利用指针的移动来遍历字符串里的所有值:
char s[] = "yangyinizhegedashabi";
char *p = s;
for (; *p!='\0'; p++) {
printf("%c\n", *p);
}
就会输出:
y
a
n
g
y
i
n
i
z
h
e
g
e
d
a
s
h
a
b
i
直接给指针赋值成字符串
上面的是指针间接的指向一个字符串,也可以这样,直接把字符串赋值给一个指针:
char *p = "mj";
int len = strlen(p);
printf("字符串长度:%d", len); //2
上面是直接把一个字符串赋值给一个指针。strlen()函数是求一个字符串的长度,参数是一个字符串,但是我们这里传入的是指针,说明,指针和字符串是可呼唤的。
我们看这个函数的定义:
size_t strlen(const char *);
它的参数时一个指针常量。其实,我们传入一个字符串的名字,其实也就是传入了这个字符串的首地址。
我们再看printf函数的定义:
printf(const char *restrict, ...)
看。参数都是一个指针字符常量。所以,可以直接定义指针。
所以总结下:
通过指针方式定义的字符串,是一个字符串常量,所以不要再给它赋值什么的都是错误的:
char *s = "mj";
*s = "like";
分析一下:上面的犯了2个容易搞混淆的错误。1是 (*s)表示字符串“mj”的首元素的地址,也就是'm'的地址,占用的空间,只占1个字节,你用"like"去填充1个字节,益处了,报错。
2是,用char *s = "mj"这种方式定义的字符串,其实是常量了,s 指向一个常量,常量是不可改变的。也会报错。
所以可以看出:
char a[] = "lmj";定义的是一个字符串变量!
char *p2 = "lmj";定义的是一个字符串常量!