[学习条记]C语言中关于指针的详解1

鼠扑  金牌会员 | 2024-8-23 21:20:50 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 870|帖子 870|积分 2610

什么是指针

操作体系给每个存储单位分(即一个字节)配了一个编号,从0x00 00 00 00 到 0xff ff ff ff
这个编号称之为地址
指针就是这个地址
c语言中 占多个多字节的变量 存储单位编号最小的那个编号 是多字节变量的地址编号
指针变量

指针变量是一个只用来存储地址的变量 但不可随意赋值一个0x00 00 00 01之类的值
因为这个地址可能还没有开发内存空间 指针变量只能生存开发好空间的地址
在32位的体系下 地址总线是32位的 所以地址编号是32位的
所以指针变量是32位的 即4个字节
指针的范例

无论什么范例的指针变量 存储的都是地址 且都是4个字节(32位体系)
对应范例的指针变量 只能存放对应范例的变量的地址
1.字符指针 char *p
2.短整形指针 short *p
3.整形指针 int p
4.长整形指针 long
p
5.浮点型指针 float *p
6.double型指针 double *p
7.函数指针
8.结构体指针
9.数组指针
10.指针的指针
指针的定义

数据范例 * 变量名;
例: int *p;
定义变量的时候 *用来修饰变量 代表p是一个指针变量
指针的运算符

  1. 在定义时 * 标识这是一个指针变量  在其他时候表示取值
  2. &  表示取一个变量的地址
  3. 例:
  4. int a = 10;
  5. int *p;
  6. 定义一个整形指针变量 p 和一个 整形变量a 值为10
  7. p = &a;   
  8. 将a的地址给p
  9. cout<<*p;
  10. 此时输出p的值 为10
  11.                
复制代码
数组的指针

  1. c语言规定 数字的名字就是数组的首地址 即第0个元素的地址,是个常量
  2.     int a[10];
  3.     a[0]=1;
  4.     a[1]=2;
  5.     int *p=a;
  6.     printf("数组地址:%p 值地址:%p 值%d\n",a,&a[1],a[1]);
  7.     printf("数组地址:%p 值地址:%p 值%d\n",p,&p[1],p[1]);
  8. 输出结果:
  9.         数组地址:0x7ff7bce31940 值地址:0x7ff7bce31944 值2
  10.         数组地址:0x7ff7bce31940 值地址:0x7ff7bce31944 值2
复制代码
注意 p和a不同, p是指针变量 而a是一个常量 是数组的首地址
例如 指针变量p可以进行 p++ 自增运算 但数组名a就不可以
a是数组的名字,是a[0]的地址,p=a即p也生存了a[0]的地址,即a和p都指向a[0] 所以在引用数组元素的时候a和p等价
不同的是 a是一个常量 存储的是a[0]的地址 而p是一个变量 所以a不可以用=赋值 p可以赋值
对a取地址和对p取地址 效果不同 对a取地址 效果为数组指针 而对p取地址 效果是指针的指针
指针的运算

  1. 指针可以加一个整数,结果还是个地址
  2.     int a[10] = {1,2,3,4,5};
  3.     int *p=&a[1];   //将数组a中第1个元素的地址给p   此时p的值为2
  4.     printf("数组地址:%p 值地址:%p 值%d\n",p,&p[0],*p);
  5.     p++;     //将指针自增1 指向下一个地址   即a[2]
  6.     printf("数组地址:%p 值地址:%p 值%d\n",p,&p[0],*p);
  7. 输出结果:
  8.         数组地址:0x7ff7bbd0b944 值地址:0x7ff7bbd0b944 值2
  9.         数组地址:0x7ff7bbd0b948 值地址:0x7ff7bbd0b948 值3
  10.        
  11.    字符数组也可以这样操作
  12.     char a[10]= "hello";
  13.     char *p = a;
  14.     printf("数组地址:%p 值地址:%p 值: %c\n",p,&p[0],*p);
  15.     p += 2;    //指向后面第二个地址 即a[2]
  16.     printf("数组地址:%p 值地址:%p 值: %c\n",p,&p[0],*p);
  17. 输出结果:
  18.         数组地址:0x7ff7bc3e095e 值地址:0x7ff7bc3e095e 值: h
  19.         数组地址:0x7ff7bc3e0960 值地址:0x7ff7bc3e0960 值: l
复制代码
指针之前可以比较巨细 条件是两个相同范例的指针指向同一个数组的元素的时候,比较巨细才故意义。
指向前面元素的指针 小于 指向后面元素的指针
两个相同范例的指针可以相互赋值 (void *范例的除外)
指针数组

  1. 一个数组中所有的元素都是类型相同的指针变量,这个数组就是指针数组
  2. 指针数组的定义:
  3.         int *p[10];     //定义一个有10个元素的数组指针 每个元素都是int类型的指针变量
  4.         和普通指针变量类似 数组指针也有不同类型 例如float double long 结构体等指针数组
  5.         一般情况下 指针数组用来保存多个字符串
  6.             char *p[3] = {"How","dare","you"};
  7.             int i = 3;
  8.             for(i=0;i<3;i++){
  9.                 printf("%s ",p[i]);
  10.             }
  11.         输出结果:
  12.                 How dare you
复制代码
指针的指针

  1. 指针的指针 即指针的地址
  2. 一个指针变量本身占4个字节 指针变量也有地址编号
  3. int a;  //定义一个变量a
  4. int *p = &a; //定义一个指针变量p 将a的地址给p
  5. 那么此时 *p就等于a
  6. int **p2 = &p; // 定义一个指针变量p2(二级指针) 将p的地址给p2
  7. 那么此时 *p2就等于p  **p2就等于a
  8. int ***p3 = &p2 定义一个指针变量p3(三级指针)同理 ***p3等价于a
  9. p p2 p3都是指针变量 都占4个字节
  10. 上代码帮助理解:
  11.     int a = 10;
  12.     int *p = &a;
  13.     int **p2 = &p;
  14.     int ***p3 = &p2;
  15.     printf("a=%d\n*p=%d\n**p2=%d\n***p3=%d\n",a,*p,**p2,***p3);
  16.     printf("&a=%p\n&p=%p\n&p2=%p\n&p3=%p\n",&a,p,p2,p3);
  17.     printf("&a=%p\n&p=%p\n&*p2=%p\n&**p3=%p\n",&a,p,*p2,**p3);
  18. 输出结果:
  19.         a=10
  20.         *p=10
  21.         **p2=10
  22.         ***p3=10
  23.         &a=0x7ff7bc836968
  24.         &p=0x7ff7bc836968
  25.         &p2=0x7ff7bc836960
  26.         &p3=0x7ff7bc836958
  27.         &a=0x7ff7bc836968
  28.         &p=0x7ff7bc836968
  29.         &*p2=0x7ff7bc836968
  30.         &**p3=0x7ff7bc836968
复制代码
字符串和指针

  1. 字符串的概念: 字符串就是以'\0'为结尾的若干字符的集合
  2. 字符串储存的形式有:数组、字符串指针、堆
  3. 字符串数组:
  4.         char str[10] = "hello";
  5.         定义一个字符数组 str  字符串hello存放在变量str中 可按下标取值
  6. 字符串指针:
  7.         char *str = "hello";
  8.         定义一个字符串指针变量 str  字符串指针变量只能存放字符地址编号
  9.         即 hello这个字符串不存放在str中  str只存放了 h 的地址编号 hello存放在文字常量区
  10. 堆:
  11.         char *str = (char*)malloc(10*sizeof(char));
  12.         动态申请了10个字节的空间,首地址给str赋值
复制代码
1.栈和全局内存中的内容是可以修改的 例如字符串数组 str[0]=“p”; 可以修改
2.文字常量区的内容是不可修改的 字符串指针 *str = “haha”;则不可以修改
3.堆区的内容是可以修改的 堆 *str =“haha”; 则可以修改
字符数组,指针定义的str可以在定义时直接初始化赋值,堆中存放的字符串不能初始化,只能通过strcpy,scanf赋值
数组指针

  1. 数组指针的概念: 本身是一个指针,指向一个数组,+1跳一个数组,即指向下一个数组
  2. 数组指针的作用就是可以保存二维数组的首地址
  3. 数组指针的定方法:指向的数组的类型+(*指针变量名)[指向的数组的元素个数]
  4.         int(*p)[5];   //定义了一个数组指针变量p  p指向的是有5个整型元素的数组
  5. 数组指针对于二维数组的应用:
  6.         int a[3][5];  //定义一个3行5列的二维数组
  7.     int (*p)[5];  //定义一个数组指针 p+1会指向下一个有5个整型元素的数组
  8.     printf("a[0]的地址: %p  a[1]的地址: %p\n",&a[0],&a[1]);
  9.     p=a;
  10.     printf("p的地址: %p  p+1的地址: %p",p,p+1);   //p+1指向下一个元素 即a[1]
  11. 输出结果:
  12.         a[0]的地址: 0x7ff7b1cfd920  a[1]的地址: 0x7ff7b1cfd934
  13.         p的地址: 0x7ff7b1cfd920  p+1的地址: 0x7ff7b1cfd934
  14. 数组指针可以将二维数组的首地址传递到另一个函数中,此时函数的形参就要定义为数组指针
  15. 一维数组指针就是二维数组的首地址 二维数组指针就是三维数组的首地址 以此类推
  16. 多维数组的数组名取*代表指针降级
复制代码
指针与函数的关系

  1. 函数的形参有int long char float等,除此之外也可以给函数传递一个地址
  2. 函数的传参方式:
  3.         1.复制传参
  4.         2.地址传参
  5.         3.全局传参(几乎用不到 全局参数可以直接使用)
  6. 定义两个函数: f1为值传递   f2为指针传递 都进行重新赋值操作
  7.         void f1(int a){
  8.            int b = 10;
  9.             a = b;
  10.             printf("%p,%d\n",&a,a);
  11.         }
  12.         void f2(int *a){
  13.             int b = 10;
  14.             *a = b;
  15.             printf("%p,%d\n",a,*a);
  16.         }
  17. 运行定义好的两个函数:
  18.         int a = 100;
  19.     f1(a);
  20.     printf("%p,%d\n",&a,a);       //传值
  21.     int a2 = 100;
  22.     f2(&a2);
  23.     printf("%p,%d\n",&a2,a2);     //传地址
  24. 输出结果:
  25.         0x7ff7b315e94c,10
  26.         0x7ff7b315e968,100
  27.         0x7ff7b315e964,10
  28.         0x7ff7b315e964,10
  29. 值传递 在函数内对形参的任何操作 不会影响实参
  30. 地址传参:将实参的地址传递给形参,形参对保存的地址的内容进行操作,实参的值也会跟着变
  31. 地址传参若要改变参数的值 必须通过 *+地址去赋值 无论这个变量是什么类型
  32. 若要传递的实参本身是一个地址,例如字符串指针,则要传递指针的地址
  33. 例:
  34.         int *str = "hello";
  35.         f2(**p);
  36. 传递数组:
  37.         传一维数组,由于数组名本身是一个首地址,所以传递时直接使用地址传递
  38.         int a[10];
  39.         f2(int*a);
  40.         传二维数组,可以使用数组指针来传递
  41.         int a[3][5];
  42.         f2(int (*a)[5]);  
  43.         传指针数组,可传指针数组的首地址,即传一个二级指针;
  44.         char *p[3] = {"what","the","fuck"};
  45.         f2(char **p);  
  46. 数组传参本质都是地址传递,所以在函数内部操作形参 都会改变实参
复制代码
准备转行嵌入式,这阵子都是特长在记条记,突然发觉手记的万一有改动不太好弄,且日后也不太好找出来看,文章都是个人心血来潮入坑嵌入式的学习条记,下一篇c语言中关于指针的详解2 会继续记载指针与函数的关系以及结构体指针,文件指针
本日累了 先溜了~



免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

鼠扑

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表