C语言-地址与指针
- 格式:ppt
- 大小:150.00 KB
- 文档页数:21
C语⾔教材第七章指针实验第七章指针第⼀部分知识训练【知识要点】1. 地址与指针指针变量的概念:⽤⼀个变量专门来存放另⼀个变量的地址。
2. 指向变量的指针变量指针变量的形式:基类型 *指针变量名;&: 取地址运算符;*:指针运算符。
3. 指针与数组引⽤⼀个数组元素可以⽤下标法和指针法;可以⽤数组名作函数的参数。
4.指针与函数函数指针变量定义的⼀般形式为:类型说明符 (*指针变量名)();⼀个函数的返回值可以是⼀个函数。
【典型习题与解析】5. char * const p;char const * pconst char *p上述三个有什么区别?【解析】char * const p; //常量指针,p的值不可以修改;char const * p;//指向常量的指针,指向的常量值不可以改;const char *p; //和char const *p。
6. main(){int a[5]={1,2,3,4,5};int *ptr=(int *)(&a+1);printf("%d,%d",*(a+1),*(ptr-1));}输出结果是什么?【解析】答案:输出:2,5*(a+1)就是a[1],*(ptr-1)就是a[4],执⾏结果是2,5。
&a+1不是⾸地址+1,系统会认为加⼀个a数组的偏移,是偏移了⼀个数组的⼤⼩(本例是5个int)。
1)&a是数组指针,其类型为 int (*)[5];⽽指针加1要根据指针类型加上⼀定的值,不同类型的指针+1之后增加的⼤⼩不同。
2)a是长度为5的int数组指针,所以要加 5*sizeof(int),所以ptr实际是a[5],但是prt与(&a+1)类型是不⼀样的(这点很重要),所以prt-1只会减去sizeof(int*) a,&a的地址是⼀样的,但意思不⼀样。
3)a是数组⾸地址,也就是a[0]的地址,&a是对象(数组)⾸地址,a+1是数组下⼀元素的地址,即a[1],&a+1是下⼀个对象的地址,即a[5]。
c语⾔指针与地址的区别
指针由两部分组成,指针的类型和指针的值(也就是变量的地址)。
指针和地址的区别:
地址只是⼀堆⼗六进制的字符,对应着内存条的某段内存,⽽指针本⾝有地址,指针的值也是⼀个地址,指针本⾝还有类型,这与单纯的地址是不同的。
指针和地址的联系:
地址可以强转成⼀个指针,例如:
int a = 1;
//假设a的地址是0x7dfe88
int *p = (int *)0x7dfe88;
附:指针类型的作⽤:
指针的值只是存储了某个变量的⾸地址,但是变量是有类型的,⽐如char是1个字节,int是4个字节,单纯的知道某个变量的⾸地址并⽆法完整的获取整个变量的值,必须知道从⾸地址往下读取多少个字节,指针的类型标明了从⾸地址往下读取多少个字节。
C语言指针详解1 程序如何运行当我们打开电脑中的任何一个程序运行时,我们的操作系统会将该程序存在硬盘的所有数据装载到内存中,然后有CPU 进行读取内存中的数据并进行计算,并将计算的结果返回给我们的操作系统,然后操作系统将相应的动作交付给相应的硬件来完成。
如:将声音数据交给声卡,最后有音响输出来,将图像交给显卡最后有显示器输出……但是还会有一部分数据会返回给内存,以供程序下面的语句继续使用。
我们都知道内存的容量有很大,如:4G,8G, 16G,有时候我们会打开很多的程序,所有的程序的数据都存放到我们的内存中,那么CPU是如何正确的读取我们的不同程序的数据并加以计算的哪?2 内存的假设设计为了让我们的CPU 可以很好的读取内存中的数据,内存必须做优化设计,于是给内存设定了集合设计,将我们的内存分成很多大小相同的方格(盒子),所有的数据将放入这些小盒子中,将不同的程序的数据放入到不同的小盒子中,这样就出现的模块化的内存,当我执行程序的一个命令时,CPU就会从相应的盒子读数据然后计算,由于我们硬件所能访问或计算的最小单位是字节,所以内存中的这样的一个小盒子的大小就给他规定一个字节。
3 地址和指针一般我们声明一块内存空间的时候,会给他取一个名字,为的是我们在编写程序的时候方便使用空间中存放的值,但是CPU 读数据的时候会忽视这个名字,因为CPU无法理解这样的数据,CPU 只能执行0,1代码,那么CPU是如何知道从什么地方读取数据,又到什么地方地址数据的读取的那,所以必须对内存做2次设计,就是将内存中分成的很多小盒子下面标注一些顺序的序号,例如:从第一个盒子开始,标注1,2,3,4,5,6,7,……每一个数字对应一个盒子,但是真正的内存如中不是使用这些十进制数字的,而是使用16进制整数表示的,如0x16ffee。
这些我们标记的数字就叫做内存中的地址。
由于这些地址和盒子是对应的关系,所以只要知道了地址,就可以得到对应盒子中存放的数据了,形象的说,我们说这个地址指向对应的盒子,在C语言中可以通过地址得到对应盒子的数据是*地址。
C语⾔参数传递(值传递、地址传递)+⼆级指针参数传递C语⾔参数传递⼀般分为:值传递和地址传递(本质上只有值传递)(注意:C语⾔中没有引⽤传递,C++才有引⽤传递,因为很多C语⾔环境是⽤C++编译器编译,使得C看起来⽀持引⽤传递,导致很多⽹上很多blog都把引⽤传递归为C语⾔参数传递的⼀类,当使⽤Microsoft Visual C++ 2010 Express或VC6之类的编译器时使⽤引⽤传递就⽆法编译通过)值传递:形参是实参的拷贝,改变形参的值并不会影响外部实参的值。
从被调⽤函数的⾓度来说,值传递是单向的(实参->形参)1 #include <stdio.h>23void swap(int x, int y);45 main()6 {7int a = 10, b = 20;89 swap(a, b);10 printf("a=%d\nb=%d\n", a, b);1112return0;13 }14void swap(int x, int y)15 {16int t;1718 t = x;19 x = y;20 y = t;21 }上述代码运⾏后a、b的值并未改变地址传递:形参为指向实参地址的指针,当对形参的指向操作时,就相当于对实参本⾝进⾏的操作1 #include <stdio.h>23void swap(int *x, int *y);45 main()6 {7int a = 10, b = 20;89 swap(&a, &b);10 printf("a=%d\nb=%d\n", a, b);1112return0;13 }14void swap(int *x, int *y)15 {16int t;1718 t = *x;19 *x = *y;20 *y = t;21 }上述代码执⾏后a和b值交换,a=20、b=10易错点补充:1 #include <stdio.h>23void swap(int *x, int *y);45 main()6 {7int a = 10, b = 20;8int *pp = &a;9int *kk = &b;10 swap(pp, kk);11 printf("a=%d\nb=%d\n", *pp, *kk);1213return0;14 }15void swap(int *x, int *y)16 {17int *t;1819 t = x;20 x = y;21 y = t;22 }请读者想⼀下,上述代码执⾏后a和b的值是否交换,为什么?上述代码看起来像交换了指针pp和kk的指向,实际上并没有代码的运⾏结果a=10、b=20,运⾏结果是a和b的值并没有改变,因为这时使⽤的实参pp,kk是值传递,传递的是指针的值,以指针pp来说,指针的值是变量a的地址,指针的值传⼊后⽤形参int *x和int *y接收,这⾥x和pp虽然都指向变量a的地址,但指针x和pp⾃⾝的地址并不相同(意思是x拷贝了⼀份pp的值),意味着你能改变变量a的值,但是不能改变pp的值(这⾥与值传递相似)为了更加直观,清晰的看出值的交换,这⾥添加⼀些代码来显⽰内存和变量的值1 #include <stdio.h>23void swap(int *x, int *y);45 main()6 {7int a = 10, b = 20;8int *pp = &a;9int *kk = &b;1011 printf("a的地址%p----b的地址%p\n\n", &a, &b);12 printf("pp的值%p----kk的值%p\n", pp, kk);13 printf("pp的地址%p----kk的地址%p\n\n", &pp, &kk);14 swap(pp, kk);15 printf("a = %d\nb = %d", *pp, *kk);1617return0;18 }19void swap(int *x, int *y)20 {21int *t;2223 printf("x的值%p----y的值%p\n", x, y);24 printf("x的地址%p----y的地址%p\n", &x, &y);26 t = x;27 x = y;28 y = t;29 }从pp和x的地址可以看出,x和pp是两块不同的内存区域,x在swap函数内执⾏完后并不会对pp值产⽣任何影响,相当于复制了⼀份pp的值,如下图(灵魂画⼿已上线)传⼊指针的值虽然不能对指针的值进⾏修改,但是可以通过地址直接对a的值进⾏修改也可以实现交换,代码如下1 #include <stdio.h>23void swap(int *x, int *y);45 main()6 {7int a = 10, b = 20;8int *pp = &a;9int *kk = &b;1011 swap(pp, kk);12 printf("a = %d\nb = %d", *pp, *kk);1314return0;15 }16void swap(int *x, int *y)17 {18int t;1920 t = *x;21 *x = *y;22 *y = t;23 }传⼊指针的值和和传⼊变量的地址在数值上是⼀样的,但是⼀个是传值传递⼀个地址传递,如下图(灵魂画⼿已经上线)⼆级指针上⾯提到的参数传递⽅式有,传⼊变量的值,传⼊变量的地址,传⼊指针的值三种参数传递⽅式下⾯介绍第四种,传⼊指针的地址,上⾯第⼆种(传⼊变量的地址)和第三种(传⼊指针的值)传递⽅式使⽤的⽤来接收参数的形参都是int *类型的指针这⾥因为传⼊的是指针的地址,所以要使⽤⼆级指针int **x,因为传⼊的指针的地址,实现值的交换的⽅式就有两种⼀种是直接对变量的值进⾏修改,另⼀种就是对指针的值进⾏修改第⼀种:直接对变量值进⾏修改1 #include <stdio.h>23void swap(int **x, int **y);45 main()6 {7int a = 10, b = 20;8int *pp = &a;9int *kk = &b;11 swap(&pp, &kk);12 printf("a = %d\nb = %d", *pp, *kk);1314return0;15 }16void swap(int **x, int **y)17 {18int t;1920 t = **x;21 **x = **y;22 **y = t;23 }第⼆种:对指针的值进⾏修改1 #include <stdio.h>23void swap(int **x, int **y);45 main()6 {7int a = 10, b = 20;8int *pp = &a;9int *kk = &b;1011 swap(&pp, &kk);12 printf("a = %d\nb = %d", *pp, *kk);1314return0;15 }16void swap(int **x, int **y)17 {18int *t;1920 t = *x;21 *x = *y;22 *y = t;23 }注意swap函数⾥⽤int *来交换指针的值,⽽不能⽤int **来进⾏交换,读者可⾃⾏修改代码验证想⼀下为什么?下⾯是⼆级指针⽰意图(灵魂画⼿已上线)若有疑问,欢迎留⾔。
c语言指针的用法c语言是一种高级编程语言,它可以直接操作内存中的数据。
指针是c语言中一种特殊的变量,它可以存储另一个变量的地址,也就是内存中的位置。
通过指针,我们可以间接地访问或修改内存中的数据,从而实现更高效和灵活的编程。
本文将介绍c语言指针的基本概念、定义和初始化、运算和应用,以及一些常见的错误和注意事项。
希望本文能够帮助你掌握c语言指针的用法,提高你的编程水平。
指针的基本概念指针是一种数据类型,它可以存储一个地址值,也就是内存中某个位置的编号。
每个变量在内存中都有一个唯一的地址,我们可以用指针来记录这个地址,然后通过这个地址来访问或修改变量的值。
例如,假设有一个整型变量a,它的值为10,它在内存中的地址为1000(为了简化,我们假设地址是十进制数)。
我们可以定义一个指向整型的指针p,并把a的地址赋给p,如下所示:int a =10; // 定义一个整型变量a,赋值为10int*p; // 定义一个指向整型的指针pp =&a; // 把a的地址赋给p这里,&a表示取a的地址,也就是1000。
p = &a表示把1000赋给p,也就是让p指向a。
从图中可以看出,p和a是两个不同的变量,它们占用不同的内存空间。
p存储了a的地址,也就是1000。
我们可以通过p 来间接地访问或修改a的值。
指针的定义和初始化指针是一种数据类型,它需要在使用前进行定义和初始化。
定义指针时,需要指定它所指向的变量的类型。
初始化指针时,需要给它赋一个有效的地址值。
定义指针的一般格式为:type *pointer_name;其中,type表示指针所指向的变量的类型,如int、char、float等;pointer_name表示指针的名称,如p、q、ptr等;*表示这是一个指针类型。
例如:int*p; // 定义一个指向整型的指针pchar*q; // 定义一个指向字符型的指针qfloat*ptr; // 定义一个指向浮点型的指针ptr注意,在定义多个指针时,每个指针前都要加*号,不能省略。
指针一、指针的概念指针即地址,一个变量的指针就是指该变量的地址。
注意:指针变量中只能存放地址。
二、指针变量的定义和引用1、指针变量的定义int *p; 此语句定义了一个指针变量p,p中可存放一个整型变量的地址。
注意:①*是指针变量的特征②只是分配了一个存储单元,并没有指真正指向,要想使一个指针变量指向一个整型变量必须赋值。
例如::int *p,I=3;p=&I;如果p=3就是错误的。
2、指针变量的引用(两个有关指针的运算符)①& 取地址运算符号②* 指针运算符*p表示p所指向的变量值。
int *p,a;p=&a; /*指向变量a的值赋给p*/scanf(“%d”,p);/*从键盘输入一个值赋值给p所指向的变量a*/*p=5; /*把5赋值给变量p所指向的a*/三、指针作为函数参数函数的参数不仅可以是整型、实型、字符型等数据,还可以是指针类型,它的作用是将一个变量的地址传送到另一个函数中四、指针与数组1、一维数组的指针表示方法(1)数组中各元素的地址。
int a[10]={1,2,3,4,5,6,7,8,9,10};①&a[0] &a[1] &a[2] 、、、&a[i]、、、&a[9]②a a+1 a+2 、、、a+i、、、、、a+9(2)数组元素值①a[0] a[1] a[2] 、、、、a[i]、、、、a[9]②*(a+0) *(a+1) *(a+2)、、*(a+i) *(a+9)2、二维数组的指针表示方法例:int a[3][4]={1,2,3,4,5,6,7,8,9,10,11,12};1、每行的起始地址①&a[0][0] &a[1][0] &a[2][0]②a[0] a[1] a[2]③a+0 a+1 a+2④*a *(a+1) *(a+2)⑤&a[0] &a[1] &a[2]2、各元素的地址①&a[0][0] &a[0][1] &a[0][2]②a[0]+1 a[0]+2 a[1]+2③*a+1 *(a+1)+1 *(a+2)+23、各元素的值①*(&a[0][0]) *(&a[0][1]) *(&a[0][2])②*(a[0]+1) *(a[0]+2) *(a[1]+2)③*(*a+1) *(*(a+1)+1) *(*(a+2)+2)四、指针与数组;printf(“%d”,*p);注意:int (*p)[5]表示p是指针变量,它指向一个包含5个元素的一维数组int *p[5] 是指针数组。
c语言中的指针是什么很多学习C语言的新手来说,指针无疑是一个难点。
但是,我觉得指针也是C语言特别重要的一个特性。
那么下面一起来看看店铺为大家精心推荐的c语言中的指针是什么,希望能够对您有所帮助。
为什么说指针是 C 语言的精髓?“指”是什么意思?其实完全可以理解为指示的意思。
比如,有一个物体,我们称之为A。
正是这个物体,有了这么个称谓,我们才能够进行脱离这个物体的实体而进行一系列的交流。
将一个物体的指示,是对这个物体的抽象。
有了这种抽象能力,才有所谓的智慧和文明。
所以这就是“指示”这种抽象方法的威力。
退化到C语言的指针,指针是一段数据/指令(在冯诺易曼体系中,二者是相通,在同一空间中的)的指示。
这是指示,也就是这段数据/指令的起始位置。
但是数据/代码是需要一个解释的方法的。
比如0x0001,可以作为一个整数,也可以作为作为一串指令,也可以作为一串字符,总之怎样解释都可以。
而C语言,在编译阶段,确定了这段数据/指令的“解释方法”。
例如,整型指针,表示的就是可以从这个指针p指向的位置开始解释,解释为一个整数。
一个函数指针,表示的就是可以从这个指针p指向的位置开始解释,解释为一段指令,对应的输入和输出以及返回值按照函数指针的类型,符合相应的要求。
综上,C语言的精髓是指针,但指针不仅仅是C语言的精髓,它是抽象的精髓。
各个语言中都有类似的东西,例如函数,例如引用。
(引用和指针的区别,我的理解,不可以进行+/-偏移操作的指针,就是引用。
随意偏移,很容易使得目标位置不符合其相应的意义,从而造成解释失败,进而崩溃。
而增加了偏移功能的指针,好处是方便表述一堆具有相同类型的数据/指令,数组之类的就是这样的实例。
) 同样的void类型的指针,也是C语言的特色。
void型的指针,就是去掉了指定类型的指针,从而使得可以以任意解释方式,解释指针,这就带来了如上的潜在问题。
但是也可以说,这个C语言的特有威力(我一般都把C语言的威力理解为这个)。
c语⾔中函数的参数、对象、地址、取址运算符、指针运算符。
1、c语⾔中函数的参数01、c语⾔中实参向形参的传递过程是单向传递的,在对形参进⾏修改后的值不能返回给实参。
02、函数返回调⽤源的返回值只能有⼀个。
例、求两个数的和与差。
#include <stdio.h>void sum_diff(int x, int y, int sum, int diff){sum = x + y;diff = (x > y) ? (x - y) : (y - x);}int main(void){int a, b, wa = 0, sa = 0;puts("please input two integers.");printf("a = "); scanf("%d", &a);printf("b = "); scanf("%d", &b);sum_diff(a, b, wa, sa); //wa,sa作为实参传递给形参sum和diff,sum和diff在调⽤函数时重新进⾏了赋值,但是并不能返回给函数的调⽤位置main函数中的变量。
(实参→形参,单向复制) // wa和sa的值仍然是在main函数中的值0.printf("\nwa: %d || sa: %d\n", wa, sa);return0;}为了解决这个问题(对传⼊函数中的变量的值进⾏修改,并返回调⽤源),需要c语⾔中的指针。
2、对象和地址001、对象是不是就是变量002、变量拥有多种属性:数据类型、存储期、变量名等。
003、对象:继承了变量属性⽽创建的变量实体称为对象。
004、地址:指的是对象的地址。
对象在内存上的存储位置编号。
本质:存储位置编号(指针)。
3、取址运算符001、取址运算符:单⽬运算符 &称为取址运算符。