部队网站建设,网页设计师培训有哪些机构,wordpress 博客摘要,网站不备案怎么回事由于指针的灵活性#xff0c;导致指针能代替数组使用#xff0c;或者混合使用#xff0c;这些导致了许多指针和数组的迷惑#xff0c;因此#xff0c;刻意再次深入探究了指针和数组这玩意儿#xff0c;其他类型的数组比较简单#xff0c;容易混淆的是字符数组和字符指针…由于指针的灵活性导致指针能代替数组使用或者混合使用这些导致了许多指针和数组的迷惑因此刻意再次深入探究了指针和数组这玩意儿其他类型的数组比较简单容易混淆的是字符数组和字符指针这两个。。。下面就开始剖析一下这两位的恩怨情仇。。。 数组的本质 数组是多个元素的集合在内存中分布在地址相连的单元中所以可以通过其下标访问不同单元的元素。。 指针。 指针也是一种变量只不过它的内存单元中保存的是一个标识其他位置的地址。。由于地址也是整数在位平台下指针默认为位。。 指针的指向 指向的直接意思就是指针变量所保存的其他的地址单元中所存放的数据类型。 int * p ;//p 变量保存的地址所在内存单元中的数据类型为整型 float *q;// ........................................浮点型 不论指向的数据类型为那种指针变量其本身永远为整型因为它保存的地址。 4 字符数组。。。 字面意思是数组数组中的元素是字符。。确实这就是它的本质意义。 char str[10]; 定义了一个有十个元素的数组元素类型为字符。 C语言中定义一个变量时可以初始化。 char str[10] {hello world}; 当编译器遇到这句时会把str数组中从第一个元素把hello world\0 逐个填入。。 由于C语言中没有真正的字符串类型可以通过字符数组表示字符串因为它的元素地址是连续的这就足够了。 C语言中规定数组代表数组所在内存位置的首地址也是 str[0]的地址即str str[0]; 而printf(%s,str); 为什么用首地址就可以输出字符串。。 因为还有一个关键在C语言中字符串常量的本质表示其实是一个地址这是许多初学者比较难理解的问题。。。 举例 char *s ; s China; 为什么可以把一个字符串赋给一个指针变量。。 这不是类型不一致吗 这就是上面提到的关键 。。 C语言中编译器会给字符串常量分配地址如果 China, 存储在内存中的 0x3000 0x3001 0x3002 0x3003 0x3004 0x3005 . s China 意识是什么对了地址。 其实真正的意义是 s China 0x3000; 看清楚了吧 你把China 看作是字符串但是编译器把它看作是地址 0x3000即字符串常量的本质表现是代表它的第一个字符的地址。。。。。。。。。。 s 0x3000 这样写似乎更符合直观的意思。。。 搞清楚这个问题。。 那么 %s 它的原理其实也是通过字符串首地址输出字符串printf(%s , s); 传给它的其实是s所保存的字符串的地址。。。 比如 [cpp] view plaincopy print? #include stdio.h int main() { char *s; s hello; printf(%p\n,s); return 0; } 可以看到 s 0x00422020 这也是China的首地址 所以printf(%s,0x00422020);也是等效的。。 字符数组 char str[10] hello 前面已经说了str str[0] 也等于 hello的首地址。。 所以printf(%s,str); 本质也是 printf(%s, 地址); C语言中操作字符串是通过它在内存中的存储单元的首地址进行的这是字符串的终极本质。。。 5 char * 与 char a[ ]; char *s; char a[ ] ; 前面说到 a代表字符串的首地址而s 这个指针也保存字符串的地址其实首地址即第一个字符的地址这个地址单元中的数据是一个字符 这也与 s 所指向的 char 一致。 因此可以 s a; 但是不能 a s; C语言中数组名可以复制给指针表示地址 但是却不能赋给给数组名它是一个常量类型所以不能修改。。 当然也可以这样: [cpp] view plaincopy print? char a [ ] hello; char *s a; for(int i 0; i strlen(a) ; i) printf(%c, s[i]); 或 printf(%c,*s); 字符指针可以用 间接操作符 *取其内容也可以用数组的下标形式 [ ]数组名也可以用 *操作因为它本身表示一个地址 。。 比如 printf(%c,*a); 将会打印出 h char * 与 char a[ ] 的本质区别 当定义 char a[10 ] 时编译器会给数组分配十个单元每个单元的数据类型为字符。。 而定义 char *s 时 这是个指针变量只占四个字节32位用来保存一个地址。。 sizeof(a) 10 sizeof(s) ? 当然是4了编译器分配4个字节32位的空间这个空间中将要保存地址。。。 printf(%p,s); 这个表示 s 的单元中所保存的地址。。 printf(%p,s); 这个表示变量本身所在内存单元地址。。。。不要搞混了。。 用一句话来概括就是 char *s 只是一个保存字符串首地址的指针变量 char a[ ] 是许多连续的内存单元单元中的元素为char 之所以用 char *能达到 char a [ ] 的效果还是字符串的本质地址即给你一个字符串地址便可以随心所欲的操所他。。但是char* 和 char a[ ] 的本质属性是不一样的。。 6 char ** 与char * a[ ] ; 先看 char *a [ ] ; 由于[ ] 的优先级高于* 所以a先和 [ ]结合他还是一个数组数组中的元素才是char * 前面讲到char * 是一个变量保存的地址。。 所以 char *a[ ] {China,French,America,German} 同过这句可以看到 数组中的元素是字符串那么sizeof(a) 是多少呢有人会想到是五个单词的占内存中的全部字节数 6787 28 但是其实sizeof(a) 16 为什么前面已经说到 字符串常量的本质是地址a 数组中的元素为char * 指针指针变量占四个字节那么四个元素就是16个字节了 看一下实例 [cpp] view plaincopy print? #include stdio.h int main() { char *a [ ] {China,French,America,German}; printf(%p %p %p %p\n,a[0],a[1],a[2],a[3]); return 0; } 可以看到数组中的四个元素保存了四个内存地址这四个地址中就代表了四个字符串的首地址而不是字符串本身。。。 因此sizeof(a)当然是16了。。 注意这四个地址是不连续的它是编译器为China,French,America,German 分配的内存空间的地址 所以四个地址没有关联。 [cpp] view plaincopy print? #include stdio.h int main() { char *a [ ] {China,French,America,German}; printf(%p %p %p %p\n,a[0],a[1],a[2],a[3]); //数组元素中保存的地址 printf(%p %p %p %p\n,a[0],a[1],a[2],a[3]);//数组元素单元本身的地址 return 0; } 可以看到 0012FF38 0012FF3C 0012FF40 0012FF44,这四个是元素单元所在的地址每个地址相差四个字节这是由于每个元素是一个指针变量占四个字节。。。 char **s; char **为二级指针 s保存一级指针 char *的地址关于二级指针就在这里不详细讨论了 简单的说一下二级指针的易错点。 举例 [cpp] view plaincopy print? char *a [ ] {China,French,America,German}; char **s a; 为什么能把 a赋给s,因为数组名a代表数组元素内存单元的首地址即 a a[0] 0012FF38; 而 0x12FF38即 a[0]中保存的又是 00422FB8 ,这个地址 00422FB8为字符串China的首地址。 即 *s 00422FB8 China; 这样便可以通过s 操作 a 中的数据 [cpp] view plaincopy print? printf(%s,*s); printf(%s,a[0]); printf(%s,*a); 都是一样的。。。 但还是要注意不能a s前面已经说到a 是一个常量。。 再看一个易错的点 [cpp] view plaincopy print? char **s hello world; 这样是错误的 因为 s 的类型是 char ** 而 hello world 的类型是 char * 虽然都是地址 但是指向的类型不一样因此不能这样用。从其本质来分析hello world,代表一个地址比如0x003001,这个地址中的内容是 h ,为 char 型而 s 也保存一个地址 这个地址中的内容(*s) 是char * 是一个指针类型 所以两者类型是不一样的。 。。 如果是这样呢 [cpp] view plaincopy print? char **s; *s hello world; 貌似是合理的编译也没有问题但是 printf(%s,*s),就会崩溃 why?? 咱来慢慢推敲一下。。 printf(%s,*s); 时首先得有s 保存的地址再在这个地址中找到 char * 的地址即*s; 举例 [cpp] view plaincopy print? s 0x1000; 在0x1000所在的内存单元中保存了hello world的地址 0x003001 *s 0x003001; 这样printf(%s,*s); 这样会先找到 0x1000,然后找到0x003001; 如果直接 char **s; [cpp] view plaincopy print? *s hello world; s 变量中保存的是一个无效随机不可用的地址 谁也不知道它指向哪里。。。。*s 操作会崩溃。。 所以用 char **s 时要给它分配一个内存地址。 [cpp] view plaincopy print? char **s ; s (char **) malloc(sizeof(char**)); *s hello world; 这样 s 给分配了了一个可用的地址比如 s 0x412f; 然后在 0x412f所在的内存中的位置保存 hello world的值。。 再如 [cpp] view plaincopy print? #include stdio.h void buf( char **s) { *s message; } int main() { char *s ; buf(s); printf(%s\n,s); } 二级指针的简单用法。。。。说白了二级指针保存的是一级指针的地址它的类型是指针变量而一级指针保存的是指向数据所在的内存单元的地址虽然都是地址但是类型是不一样的。。。
转自http://blog.csdn.net/daiyutage/article/details/8604720