程序展示
打印一个有10个元素的数组,将其全置为1
原始方式
int main() { int arr[10] = { 0 }; int sz = sizeof(arr) / sizeof(arr[0]); int i = 0; for (i = 0; i < sz; i++) { //将数字全部置为1 arr[i] = 1; //输出 printf("%d ", arr[i]); } printf("\n"); return 0; }
指针变量方式
让我们先看一段代码
int main() { int a[10] = { 1,2,3,4,5,6,7,8,9,10 }; // 0 1 2 3 4 5 6 7 8 9 int sz = sizeof(a) / sizeof(a[0]); int* p = &a;//取a数组的地址 printf("%d\n", a[1]); printf("%d\n", *(a + 1)); printf("%d\n", *(p + 1)); return 0; }
结果会输出什么呢?
*(p+i) 或 * (a+i)是p+i 或a+i所指向的数组元素
([ ]是变址运算符,将a[i]按a+i计算地址,然后找出此地址单元的值)
小结:
可以看出,*(p+i),*(a+i),a[i]三者是等价的。
同理,我们也可以看出
*p 和 *arr 也是等价的,都是取首元素的地址
代码对比
指针运算
指针+-整数
用指针打印数组内容
int main() { int a[10] = { 1,2,3,4,5,6,7,8,9,10 }; int sz = sizeof(a) / sizeof(a[0]); int* p = &a; int i = 0; for (i = 0; i < sz; i++) { /*方法① printf("%d ", *p); p++;*/ /*方法②*/ printf("%d ", *(p + i)); } printf("\n"); return 0; }
结果打印输出
使用指针打印1-10中的奇数
int main() { int a[10] = { 1,2,3,4,5,6,7,8,9,10 }; int* p = &a; int i = 0; for (i = 0; i < 5; i++) { printf("%d ", *p); p += 2; } printf("\n"); return 0; }
结果打印输出
倒序输出数组
int main() { int a[10] = { 1,2,3,4,5,6,7,8,9,10 }; // 0 1 2 3 4 5 6 7 8 9 int sz = sizeof(a) / sizeof(a[0]); int* p = &a[9];//指向数组最后一个元素 int i = 0; for (i = 0; i < sz; i++) { printf("%d ", *(p - i)); } printf("\n"); return 0; }
结果打印输出
指针-指针
指针 - 指针 = 地址 - 地址
需要指向同一块空间
看下段代码👇
int main() { int a[5] = { 1,2,3,4,5 }; // 0 1 2 3 4 int* p1 = &a[4]; int* p2 = &a[0]; int* p3 = &a[1]; printf("%d\n", p1 - p2); printf("%d\n", p1 - p3); return 0; }
结果打印输出
小结:指针相减,代表的是指针之间的距离(用数组的元素来进行度量)
再看下段代码👇
int my_strlen(char* s) { char* p = s; while (*p != '\0') p++; return p - s; } int main() { printf("%d\n", my_strlen("abc")); return 0; }
结果打印输出为:3
解释一下这段代码:
这段代码是链表算长度和找尾节点
把字符串传过去的地址拷贝了一份叫p
p-s是:p向后走了多少就知道字符串首尾差了多少
指针的关系运算
看下段代码
int main() { int a[10] = { 1,2,3,4,5,6,7,8,9,10 }; int sz = sizeof(a) / sizeof(a[0]); int* p = &a[0]; while (p < a + sz) { printf("%d ", *p); p++; } printf("\n"); return 0; }
代码结果输出
代码解释