指针的主题,我们在初级阶段的《指针》章节已经接触过了,我们知道了指针的概念:
1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
2. 指针的大小是固定的4/8个字节(32位平台/64位平台)。
3. 指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限。
4. 指针的运算。
这个章节,我们继续探讨指针的高级主题。
首先分析以下代码:
#include<stdio.h> int main() { int a = 10; printf("%p\n", &a); return 0; }
编辑 注:上图中打印地址只供分析使用,因为在不同机器中打印的地址是不相同的!
1. 字符指针
在指针的类型中我们知道有一种指针类型为字符指针 char* ;
一般使用:
int main() { char ch = 'w'; char *pc = &ch; *pc = 'w'; return 0; }
还有一种使用方式如下:
int main()
{
const char* pstr = "hello bit"; //这里是把一个字符串放到pstr指针变量里了吗?
printf("%s\n", pstr);
return 0;
}
代码 const char* pstr = "hello bit."; 特别容易让同学以为是把字符串 hello bit 放到字符指针 pstr 里了,但是/本质是把字符串 hello bit. 首字符的地址放到了pstr中。
编辑
注:上图中打印地址只供分析使用,因为在不同机器中打印的地址是不相同的!
上面代码的意思是把一个常量字符串的首字符 h 的地址存放到指针变量 pstr 中。
再来分析以下代码:
int main() { //int a = 10; //printf("%p\n", &a); char ch = 'w'; char* p = &ch; char* p = "abcdef"; return 0; }
编辑
分析以下代码:
#include <stdio.h> int main() { char* p = "abcdef"; *p = 'w'; return 0; }
编辑 结论:以上代码逻辑错误!此处无法对p进行更改,因为abcdef是存储在内存的只读数据区!
因此,代码严谨写法如下:(添加const表示不可更改!)
#include<stdio.h> int main() { const char* p = "abcdef"; *p = 'w'; return 0; }
编辑注: 这样编写代码更加严谨!
那就有可这样的面试题:
#include <stdio.h> int main() { char str1[] = "hello bit."; char str2[] = "hello bit."; const char* str3 = "hello bit."; const char* str4 = "hello bit."; if (str1 == str2) printf("str1 and str2 are same\n"); else printf("str1 and str2 are not same\n"); if (str3 == str4) printf("str3 and str4 are same\n"); else printf("str3 and str4 are not same\n"); return 0; }
这里最终输出的是:
编辑
这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当 几个指针。指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化 不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4不同。
分析以下代码:
int main() { char arr1[] = "abcdef"; char arr2[] = "abcdef"; const char* str1 = "abcdef"; const char* str2 = "abcdef"; if (arr1 == arr2) printf("arr1==arr2\n"); else printf("arr1!=arr2\n"); if (str1 == str2) printf("str1==str2\n"); else printf("str1!=str2\n"); return 0; }
编辑 注:
1.arr1和arr2是两个不同的数组,而数组名又代表首元素的地址,这是内存中两块不同的空间,其起始地址不相同,故arr1!=arr2。
2.str1和str2为常量字符串,不能被更改,并且其存储的内容相同,因此没有必要存储两份!故str1和str2指向相同!
3.图中打印地址为假设,讲解使用,不必深究此地址内容!
注意:
编辑
编辑
编辑
编辑
编辑
2. 指针数组
在《指针》章节我们也学了指针数组,指针数组是一个存放指针的数组。
这里我们再复习一下,下面指针数组是什么意思?
int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5]; //二级字符指针的数组
编辑
编辑
编辑
int main() { char* arr[] = { "abcdef", "qwer", "zhangsan" }; int i = 0; int sz = sizeof(arr) / sizeof(arr[0]); for (i = 0; i < sz; i++) { printf("%s\n", arr[i]); } return 0; }
编辑
分析以下代码:
编辑
int main() { int arr1[] = { 1,2,3,4,5 }; int arr2[] = { 2,3,4,5,6 }; int arr3[] = { 3,4,5,6,7 }; int* arr[] = {arr1, arr2, arr3}; int i = 0; for (i = 0; i < 3; i++) { int j = 0; for (j = 0; j < 5; j++) { printf("%d ", arr[i][j]);//*(*(arr+i)+j) } printf("\n"); } return 0; }
此代码模拟了二维数组。