指针的主题,我们在初级阶段的《指针》章节已经接触过了,我们知道了指针的概念:
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; }
此代码模拟了二维数组。