4. 指针运算
指针± 整数
指针-指针
指针的关系运算
4.1 指针±整数
#include <stdio.h> int my_strlen(char * str) //模拟strlen { int count = 0; while (*str != '\0') { count++; //str++;//指针+整数 str = str + 1; } return count; } int main() { int len = my_strlen("abcdef"); printf("%d\n", len); return 0; }
另一种演示
#define N_VALUES 5 float values[N_VALUES]; float* vp; //指针+-整数;指针的关系运算 for (vp = &values[0]; vp < &values[N_VALUES];) { *vp++ = 0; }
图文讲解
4.2 指针-指针
int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int n = &arr[0] - &arr[9]; printf("%d\n", n); return 0; }
图片详解
总结:
1.两个指针指向同一块空间,指针的类型是一致的
2.指针+整数=指针,指针-指针=整数
再举一个例子(模拟strlen)
int my_strlen(char * str) { char* start = str; while (*str != '\0') str++; return str - start;//指针-指针得到的是元素个数 } int main() { int len = my_strlen("abcdef"); printf("%d\n", len); return 0; }
4.3 指针的关系运算
for(vp = &values[N_VALUES]; vp > &values[0];) { *--vp = 0; }
图片详解
代码简化, 这将代码修改如下:
for(vp = &values[N_VALUES-1]; vp >= &values[0];vp--) { *vp = 0; }
图片详解
实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。
标准规定:
允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。
5. 指针和数组
我们看一个例子:
#include <stdio.h> int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,0 }; printf("%p\n", arr); printf("%p\n", &arr[0]); return 0; }
运行结果:
可见数组名和数组首元素的地址是一样的。
结论:
数组名表示的是数组首元素的地址。
那么这样写代码是可行的:
int arr[10] = {1,2,3,4,5,6,7,8,9,0}; int *p = arr;//p存放的是数组首元素的地址
既然可以把数组名当成地址存放到一个指针中,我们使用指针来访问一个就成为可能。
指针就是指针,不是数组;数组就是数组,也不是指针。
指针的大小:4/8个字节,指针是存档地址的,地址的存放需要多大空间,指针变量的大小就是多少。
数组的大小:取决数组的元素个数和每个元素的类型。
指针可以指向数组元素的
因为指针可以运算,所以借助于指针可以访问数组
例如:
int main() { int arr[10] = { 0 }; int* p = arr;//&arr[0] int i = 0; //存放 for (i = 0; i < 10; i++) { *p = i + 1; p++; } //打印 p = arr; for (i = 0; i < 10; i++) { *(p + i) = i + 1; printf("%d ", *(p + i)); } return 0; }
代码结果:
6. 二级指针
指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?
这就是 二级指针 。
举例代码如下:
#include <stdio.h> 二级指针 int main() { int a = 10;//a是要在内存中申请4个字节的空间的 //一级指针 int* pa = &a;//0x0012ff40, pa是指针变量,用来存放地址,也得向内存申请,申请4/8 //二级指针 int** ppa = &pa;//0x0012ff48 int** * pppa = &ppa; printf("%d\n", **ppa); return 0; } //指针数组 int main() { //int* p; //int arr[10];// //char ch[5];// int a = 10; int b = 20; int c = 30; //指针数组-存放指针的数组 int* arr[] = { &a, &b, &c }; int i = 0; for (i = 0; i < 3; i++) { printf("%d ", *(arr[i])); } char* arr2[5]; float* arr3[5]; return 0; } int main() { char* arr[5];//[char* char* char* char* char*] char** p = arr;//&arr[0] - char** return 0; }
对于二级指针的运算有:
*ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是 pa .
int b = 20; *ppa = &b;//等价于 pa = &b;
**ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作: *pa ,那找到的是 a .
**ppa = 30; //等价于*pa = 30; //等价于a = 30;
7. 指针数组
指针数组是指针还是数组?
答案:是数组。是存放指针的数组。
数组我们已经知道整形数组,字符数组。
用指针数组模拟二维数组
int main() { int arr1[] = { 1,2,3,4,5 }; int arr2[] = { 2,3,4,5,6 }; int arr3[] = { 3,4,5,6,7 }; int* ptr[] = {arr1, arr2, arr3}; int i = 0; for (i = 0; i < 3; i++) { int j = 0; for (j = 0; j < 5; j++) { printf("%d ", ptr[i][j]); } printf("\n"); } return 0; }
通过代码了解指针数组
void test(int arr[], int sz) { int i = 0; for (i = 0; i < 10; i++) { arr[i] = i; } } int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int i = 0; for (i = 0; i < 10; i++) { printf("%d ", i[arr]); //i[arr] -- *(i+arr) //arr[i] -- *(arr+i) } //int* p = arr; //test(arr, 10); *(arr+i) -- *(p+i) -- arr[i] arr[i] --> *(arr+i) --> *(i+arr) -- i[arr] //for (i = 0; i < 10; i++) //{ // printf("%p = %p\n", &arr[i], p+i); //} return 0; } int main() { int arr[3][5]; //arr[i][j] //(*(arr+i))[j] //*(*(arr+i)+j) return 0; }
如果这份博客对大家有帮助,希望各位给恒川一个免费的点赞作为鼓励,并评论收藏一下,谢谢大家!!!
制作不易,如果大家有什么疑问,或给恒川的意见,欢迎评论区留言。