c语言字符指针、字符串初始化问题
之前,我在做图的存储结构的时候,用邻接矩阵来表示一张图,遇到一个段错误问题:
上面只要一输入一条边,就会出现段错误,而段错误,往往就是内存地址访问出错,比如数组越界,访问了未分配的地址,访问系统已经分配的地址,都会出错,于是我将错误定位到输入边的这个位置
上面就是我在循环之前,定义了两个字符指针,准备用来存放字符串,但是只要一scanf之后,只要访问其中某一个变量就会出现段错误,难道是说这个地址不能别访问?
于是我做了一个实验,如下操作:
直接上代码:
#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> #include <string.h> void test(char* str) { printf("str=%s ", str); if (strcmp("love", str) == 0) { printf("哈哈,字符串相等 "); } else { printf("字符串不相等 "); } } int main() { char* v1; scanf("%s", &v1); test(v1); system("pause"); return 0; }
然后就会内存地址访问错误
那我又改成如下代码:
#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> #include <string.h> void test(char* str) { printf("str=%s ", str); if (strcmp("love", str) == 0) { printf("哈哈,字符串相等 "); } else { printf("字符串不相等 "); } } int main() { test("love"); system("pause"); return 0; }
运行正常:
换句话说,问题就出在:
也就是说,不能用sacnf来初始化char*,下面我们就来分析一下:
那么就是说,char* p存放的是常量字符串的地址,那我们就不能用scanf来初始化 ,因为如果用scanf来初始化,我们根本在常量区没有空间指向。而且就算有空间指向了常量区,也不能用scanf去初始化,比如如下:
因为常量区的数据读写本来就是一旦定义了就不允许改动的
那么如果我们非要用scanf来初始化一个字符串呢,那么这里我们第一时间考虑的应该是一个字符数组,代码如下:
#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> #include <string.h> void test(char* str) { printf("str=%s ", str); if (strcmp("love", str) == 0) { printf("哈哈,字符串相等 "); } else { printf("字符串不相等 "); } } int main() { char str[10] = "pxx";//OK,并且带有 scanf("%s", str); test(str);//char*指向了一个数组空间 system("pause"); return 0; }
运行结果:
对于test来说,相当于用char* str指向了char str[10]的这片数组空间,这个时候char* str有指向了,并且可以改动,我们既可以用scanf来直接赋值。
代码改动如下:
运行结果: