指针知识点总结(C语言)

一:基础认知:
                Point_Type*   Point_Name =  & (var_Type   var)

                指针类型*  指针名 = & 变量名(变量和指针必须为同一数据类型,因为是按数据类型的不同来划分内存大小的)

               int a=12;

               int* p=NULL;

                p=&a;

                printf(*p);

                

其中:

                &(取址符):p=&a 是返回a的地址赋值给指针变量p

                *                    :*p 是访问并返回指针变量p所存储的地址(a的地址)里面的值(a的值)

                例如:把int型指针p指向int型变量var:

             其中,&(取址符)的作用是取var的地址;而指针p是一个存储变量地址的指针变量;在指针p前面加一个‘ * ’则是访问指针p存储的变量地址所在的地址存储的值(即访问var的值)    

        输出一个指针变量保存的内容(也就是地址)用  %p  :

                 

var把自己的地址赋给p,  所以p(int指针型变量)的内存地址上存储的是var变量的地址;

*p是p存储的变量(var)地址存储的值(12);

所以:p==&var  ;   

           *p==var;

图解:

例如:把一个int型变量的地址赋值给int型指针变量p(在int型指针变量p上存储一个int型变量的地址)

#include <stdio.h>
int main()
{
	int a = 12;
	int* prt = &a;
	int* p;//int型指针变量指向一个int型变量的地址(即int型指针变量存储int型变量地址)
	printf("int a的地址:%p\tint a的值:%d\n",&a,a);
	printf("int* prt的地址:%p\tint* prt的值:%p\nint* prt指向的int型变量地址上存储的int型的值:%d\n",&prt,prt,*prt);
	printf("请输入一个int型变量的地址:\n");
	scanf_s("%p",&p);//在int型指针变量p上存储一个int型变量的地址
	printf("int* p的地址:%p\tint* p的值:%p\nint* p指向的int型变量地址上存储的int型的值:%d",&p,p,*p);
	return 0;
}

 结果:

 二、指针和数组:
                int* p=NULL  (假设指针p是int型的指针,赋初始值为NULL(常量0))

        (1)p++:指针p的值加4(因为p是Int型指针,所以内存单位是4个单位),即在指针p本身的地址原本所存储的变量的地址上加4,所以p++是指指针p向后移动4个内存单位

(在int数组里面表现为访问下一个元素)

        (1)p--:指针p的值4(因为p是Int型指针,所以内存单位是4个单位),即在指针p本身的地址原本所存储的变量的地址上4,所以p++是指指针p向移动4个内存单位

(在int数组里面表现为访问一个元素)

          例如:


#include <stdio.h>
#define M 3

int main()
{
	int i = 0;
	int group[] = {12,23,34};
	int* p = group; //p的值默认为group[0]的地址,即p指向group[0]
	
	while (p <= &(group[M-1]))  //指针p的值(p所指向的变量的地址)小于等于group[2]的地址:即p的值从group[0]的地址移动到group[2]的地址,步长为4(int)
	{
		printf("\ngroup[%d]=%d\t&group[%d]=%p\tp=%p\t,*p=%d\t&p(指针变量p本身的地址)=%p", i, group[i],i,&group[i],p,*p,&p);
		p++;//指针往后移动4个存储单位(int型)
		i++;
	}

	return 0;
}

   结果:

可以看到:group[i]==*p ;&group[i]==p ;&p一直没有变

且:&group[i]-&group[i-1]==4          其中  i 属于[1,2]     (p一次移动4给内存单位)

上述代码图解:

 三、指针和函数:

        

 (1)引用调用&传值调用:

                a、当指针作为函数的参数时,实现引用调用,引用调用就相当于把你家的地址告诉了别人,那么别人就可以在你家偷东西或放东西(修改变量的存储地址上存储的数据),比如本来你家只有一个苹果,但是被A(知道你家地址的人,相当于存储了变量地址的指针变量)换成了香蕉(进行了修改操作),那你再访问你家(地址),就只能看见香蕉了(修改后的数据),所以你只能拿香蕉(修改后的数据)去进行其他操作。

                b、当一个基本类型的变量作为函数的参数时,实现传值调用,传值调用相当于赋值(a=b),把b的value的副本给a,a和b除了当前的value相等,没有其他联系。

#include <stdio.h>

void swap1(int* a,int* b)
{
	int i = 0;

	i = *a;
	*a = *b;
	*b = i;

	printf("\nswap1().a(int*)的地址:%p\tswap1().b(int*)的地址:%p\t",&a,&b);
}

void swap2(int a,int b)
{
	int i = 0;

	i = a;
	a = b;
	b = i;
	printf("\nswap2().a(int)的地址:%p\tswap2().b(int)的地址:%p\t", &a, &b);
}

int main()
{
	int a = 23, b = 48;
	printf("\na的地址:%p\tb的地址:%p",&a,&b);
	printf("\n\n\n");
	swap2(a, b);
	printf("\n传值调用:a=%d,b=%d\t",a,b);
	printf("\n\n\n");
	swap1(&a,&b);
	printf("\n引用调用:a=%d,b=%d\t",a,b);
	printf("\n\n\n");
	return 0;
}

结果: 

void swap1(int* a,int* b)图解:

四、指针的指针

                (1)概念

                int a=12;

                int* p=&a;    //p是指向int型变量的指针变量,所以p是指向变量a的指针,p指向a的地址(int型指针p存储a的地址)

                int** pp=&p;  //pp是指向int型指针的指针变量,所以pp是指针p的指针,pp指向p的地址(int型指针的指针pp存储p的地址)

               (2)例子

#include <stdio.h>

int main()
{
	int a = 12;
	int* p = &a;	//把a的地址赋值给int型指针p
	int** pp = &p;	//把指针p地址赋值给int型指针的指针pp

	printf("\na的地址:%p\ta的值:%d\np的地址:%p\tp的值:%p\npp的地址:%p\tpp的值:%p",&a,a,&p,p,&pp,pp);//输出地址用:%p

	return 0;
}

结果:

 可以看到:

                指针p存储的值(value)是a的地址(p指向a);

                指针的指针pp存储的值(value)是指针p的地址(pp指向p);

图解:

                (3)*

访问并返回一个指针(p)所指向的变量的value:*p

访问并返回一个指针的指针(pp)所指向的指针(p)所指向的变量的value:**pp

(哈哈哈哈哈,有点绕口,哈哈哈哈哈)

如下例:

#include <stdio.h>

int main()
{
	int a = 12;
	int* p = &a;	//把a的地址赋值给int型指针p
	int** pp = &p;	//把指针p地址赋值给int型指针的指针pp

	//printf("\na的地址:%p\ta的值:%d\np的地址:%p\tp的值:%p\npp的地址:%p\tpp的值:%p",&a,a,&p,p,&pp,pp);//输出地址用:%p

	printf("\n\n\na==%d\t*p==%d\t**pp==%d\n",a,*p,**pp);

	return 0;
}

结果: 

理论上来讲:

*p是根据p存储的变量(a)的地址来访问该变量(a),并返回该变量地址上存储的数据(12)

**pp:第一个‘*’是 *pp,是根据pp存储的指针变量(p)的地址来访问该指针变量(p),并返回该指针变量(p)的地址上存储的地址(a的地址);第二个‘*’是  *(*pp)相当于 *(p),即根据p存储的变量(a)的地址来访问该变量(a),并返回该变量地址上存储的数据(12)

 也可以再一次用这张图解释:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值