在我们平时写代码中常用的为三级指针及以下,多的用的比较少
int num = 999;
//定义一级指针:
int * num_p = #
//定义二级指针:
int ** num_p_p = &num_p;
//定义三级指针:
int *** num_p_p_p = &num_p_p;
printf("num_p的地址是: %p, num_p_p的地址是: %p\n", num_p, num_p_p);
printf("num_p_p的值是: %d\n", **num_p_p);
打印:
num_p的地址是: 003DFBB8, num_p_p的地址是: 003DFBAC
num_p_p的值是: 999
在c里 定义数组和java里是差别的,c里面的 []是固定地方,而 java里面可以放在变量前后
// 定义一个数组
int arr[] = {1,2,3,4};
// 遍历数组和java也不太一样,得先声明i要不能有些编译器会出现问题 再做for循环
int i = 0;
for (i = 0; i < 4; ++i) {
printf("%d\n", arr[i]);
}
// 上面输出:
// 1
// 2
// 3
// 4
数组的特殊性
- 数组本身就是一个地址,所以上面遥 arr 和 &arr相等
- 数组的第一个值的地址也是指向数组本身, &arr[0] == &arr ==arr
// 打印上面 arr相关
printf("arr = %p \n", arr);
printf("&arr = %p \n", &arr);
printf("arr[0] = %p \n", &arr[0]);
// 输出如下:
// arr = 00CFFE84
// &arr = 00CFFE84
// arr[0] = 00CFFE84
// 所以上面又得出一个数组特性
int a =100;
// int *aP = a; // 这个不可行, 会抛错
// int *arrP = arr; //这个是可行的,因为数组本身就是一个地址
// 打印数组的值
// 打印第一个值 直接取数组的值就可以 *arr
printf("打印数组的值: %d\n", *arr); //输出: 打印数组的值: 1 证实就是第一个
// 打印第二个,可以用到指针挪动法来打印
int *arrP = arr;
arrP++; // 指针挪动,+1代表移动到下一个地方
printf("打印数组的值arrP【2】: %d\n", *arrP); // 打印数组的值arrP【2】: 2
// 也可以连续挪动二个位置 比如从第二个挪动到第四个
arrP+=2;
// 再挪动到第一个
arrp-=3;
// 指针是没有索引越界这一说法
数组内存分配
跟Java一 样C里面内存也是连续分配的
int main() {
int arr[] = {1, 2, 3, 4};
int *arr_p = arr;
int i = 0;
for (i = 0; i < 4; ++i) {
printf("位置%d的值是:%d\n ", i, *(arr_p + i));
printf("位置%d的内存地址是:%p\n ", i, arr_p + i);
}
/** int 数组 每次移动4个字节
*位置0的值是:1
位置0的内存地址是:0053F85C
位置1的值是:2
位置1的内存地址是:0053F860
位置2的值是:3
位置2的内存地址是:0053F864
位置3的值是:4
位置3的内存地址是:0053F868
*/
return 0;
}
数组遍历附值
#include <stdio.h>
int main() {
int arr[4];
int * arr_p = &arr;
// 开始循环赋值
int i =0;
for(i = 0; i<sizeof(arr)/sizeof(int);i++){
*(arr_p + i) = 1000+i;
}
// 打印输出看下是否赋值成功
int j = 0;
for(j = 0; j < sizeof(arr)/sizeof(int);j++){
printf("位置%d的值是:%d\n",j,*(arr_p+j));
}
// 获取数组字节可以用 sizeof arr === sizeof(arr) 其它不能这样写
return 0;
// 输出如下:
// 位置0的值是:10001
// 位置1的值是:10002
// 位置2的值是:10003
// 位置3的值是:10004
}