一.指针是什么
1.浅谈指针
理解指针需要知道两点:
1.指针是内存中一个最小单元的编号,也就是地址。
2.我们口中所说的指针,通常指的是指针变量,是用来存放地址的变量。
总结:指针 == 地址
2.指针变量的定义
指针变量的定义跟普通变量的定义一样,需注意类型。
int a =100;
int *p; //这里的*是一个标识符,告诉系统p是一个指针变量,是用来存放地址的
p = &a;//将整型变量a的地址存在指针变量p里
printf("%d",*(&a));
printf("%d",*p); //以上两个printf里的*是运算符,表示将后面跟的内存地址中的数据“取出来”
printf("%p",&a);//通过&取出a的地址
printf("%p",p); //a的地址已存放在指针变量p里,输出p就是输出地址
二.为什么要用指针
大概了解了指针的基本知识后,我们来思考一下为什么要用指针呢?
由于指针 == 地址,所以顾名思义,我们使用指针变量时,其实就在直接操作地址。
让我们通过一道经典的题目来探究使用指针变量的意义
题目:封装一个函数,实现两个数的交换
在没学习指针之前,我们只能这样封装函数(如下)
如果知道了函数传参时,实参和形参的区别,我们不难看出,main函数里实参的data和data2与封装函数里形参的data和data2是完全不同的地址。所以即使我们在main函数里调用了封装函数,最后输出的实参依旧是原来的样子,并没有实现交换功能。
而我们学习了指针变量后,我们就打开了新世界的大门,可以直接传输地址(如下)
我们可以直接在封装函数里定义两个指针变量,在main函数传参时,我们直接将main函数里实参的地址存放到封装函数里的两个指针变量里,这时的两数交换的地址就从始至终都一样了。其实这就是我们在学习函数数组传参时的道理,将main函数里的数组名传参,传的实际上是数组的首地址。
三. 通过指针变量输出数组
在了解了指针变量的使用方法后,我们可以定义一个指针变量指向数组,通过指针偏移(地址偏移)的方式输出数组的值,直接上代码(如下)
#include <stdio.h>
int main()
{
int i;
int len;
int arr[]={1,2,3};
int *p;
p = arr; //数组名就是数组的首地址
//p = &arr[0];
len=sizeof(arr)/4;
for(i=0;i<len;i++)
{
printf("数组的%d元素是%d\n",i,*p++);
}
}
一些见怪不怪的写法:
#include<stdio.h>
int main()
{
int i;
int arr[] = {1,2,3};
int *p = arr;
printf("size of arr is:%d",sizeof(arr)); //一个元素占4个字节,3*4=12
printf("\nsize of arr is:%d",sizeof(p));//操作系统用8个字节来表示地址,p == 数组arr的首地址
printf("\nsize of pointer is:%d",sizeof(int*));//操作系统用8个字节来表示地址,指针就是地址
printf("\nsize of pointer is:%d",sizeof(char*));//操作系统用8个字节来表示地址,指针就是地址
printf("\n%d \n",*p); //等价
printf("%d \n",*arr);
for(i=0;i<3;i++)
{
printf("%d ",p[i]);
}
puts("");
for(i=0;i<3;i++)
{
printf("%d ",*(arr+i));
}
puts("");
/*for(i=0;i<3;i++)
{
printf("%d ",*arr++); //编译不过,指针常量arr不可变
}*/
for(i=0;i<3;i++)
{
printf("%d ",*p++); //编译通过,指针变量p可变
}
}
小练习,封装好初始化数组函数,输出数组函数,代码如下:
#include<stdio.h>
void initarry(int *parry,int len) //int *parry是定义了一个指针来保存地址。(此时是标识作用)
{
int i;
for(i=0;i<len;i++)
{
printf("请输入第%d个数据:",i+1);
scanf("%d",parry++); //不带*是因为scanf需要将数值输入到地址里,parry就是一个地址。
}
}
void printfarry(int *parry,int len)
{
int i;
for(i=0;i<len;i++)
{
printf("%d ",*parry++); //带*是因为printf要输出数值而不是地址,*将地址里的值读取出来。
//此时是运算作用
}
}
int main()
{
int arry[5];
int len=sizeof(arry)/4;
initarry(&arry[0],len); //两种形式都可将arry传到封装函数里,都是传送数组的首地址
printfarry(arry,len);
}