我是搞java的,今年要找工作,参加笔试就必须要准备一些C/C++、算法、数据结构、操作系统、概率论的东西,浅谈一下我作为初学者对C指针的学习和理解。
一、*p 概念:
申明一个整形指针变量:
int * pi;
pi是一个指针,pi存的是地址编号的变量,若 int *pi = i; 则pi=&i;有下面等式:&(*pi)=pi;
例:
int a,*pa;
a=10;
pa=&a;
*pa=20;
printf( “%d”, a) //输出20;通过改变指针pa所指的内容来改变a的值,其实pa=&a;操作使pa和&a指向同一个内容。
二、指针和数组:
当一个指针变量被初始化成数组名时,就说该指针变量指向了数组。如:
char str[20], *ptr;
ptr=str; //ptr被置为数组str的第一个元素的地址(ptr=&str[0]),因为数组名就是该数组的首地址,也是数组第一个元素的地址。
此时可以认为指针ptr就是数组str(反之不成立),ptr为指向数组第一个元素的指针,值得注意的是:ptr是一个可以变化的指针变量,而str是一个常数,因为数组一经被说明,数组的地址也就是固定的,因此str是不能变化的,这样原来对数组的处理都可以用指针来实现。如对数组元素的访问,既可以用下标变量访问,也可以用指针访问。
以下代码的打印结果一样
int *pa,a[]={3,4,5,6,7,3,7,4,4,6};
for (int i=0;i<=9;i++)
{
printf ( “%d”, a[i] );
printf ( “%d”, *(a+i) );
printf(“%d”, pa[i] );
printf ( “%d”, *(pa+i) );
}
指针变量也可以指向多维数组,例如,在一个三维数组中,引用元素c[i][j][k]的地址计算最终将换成:*(*(*(c+i)+j)+k)。
若有如下说明:
int a[3][4];
int *p;
p=a;
p是指向整型变量的指针;p=a使p指向整型二维数组a的首地址。*(*(p+1)+2)表示取a[1][2]的内容;*p表示取a[0][0]的内容,因为p是指向整型变量的指针;p++表示p的内容加1,即p中存放的地址增加一个整型量的字节数2,从而使p指向下一个整型量a[0][1]。公式:a[j][k]等价于*(a+(j* row_length)+k)
三、指针常量:
int const * pi与int * const pi
int * const pi是将指针pi被定义为常量指针,
int const *pi == const int *pi是将指针pi所指的内容定义为常量
注意看如下代码:
int i1=30;
int i2=40;
int * const pi=&i1;
pi=&i2; //4.注意这里,代码不能编译成功,因为上一句指针pi被定义为了常量,不能再对pi赋值;
再看下面代码:
int i1=30;
int i2=40;
int const *pi=&i1;
pi=&i2; //注意这里,pi可以在任意时候重新赋值一个新内存地址
i2=80; //5.想想看:这里能用*pi=80;来代替吗?当然不能
printf( “%d”, *pi ) ; //6. 输出是80,但是不能通过*pi来改变i2了
补充:const int * const pi=&i;// pi值不能改,也不能通过pi修改i的值。因为不管是*pi还 是pi都是const的
此处有总结:
1) 如果const 修饰在*pi前则不能改的是*pi(即不能 类似这样:*pi=50;赋值)而不是指pi.
2) 如果const 是直接写在pi前则pi不能改(即不能类似 这样:pi=&i;赋值)。
四、值传递,地址传递,引用传递
看下面三个题目
1、值传递
void Exchg1(int x, int y)
{
int tmp;
tmp=x;
x=y;
y=tmp;
printf (“x=%d,y=%d\n”,x,y)
}
void main()
{
int a=4,b=6;
Exchg1 (a,b) ; //只是将a的值赋给x,b的值赋给y,并不能通过xy来改变ab的值;
printf(“a=%d,b=%d\n”,a,b)
}
输出的结果 :
x=6, y=4
a=4, b=6
2、地址传递
Exchg2(int *px, int *py)
{
int tmp=*px;
*px=*py;
*py=tmp;
print(“*px=%d,*py=%d\n”,*px,*py);
}
main()
{
int a=4;
int b=6;
Exchg2( &a,&b); //指针px,py的值已经分别是a,b变量的地址值了。接下来,对*px,*py的操作当然也就是对a,b变量本身的操作
Print (“a=%d,b=%d\n”, a, b);
}
*px=6, *py=4
a=6, b=4
3、引用传递
Exchg3(int &x, int &y) //注意定义
{
int tmp=x;
x=y;
y=tmp;
print(“x=%d,y=%d\n”,x,y);
}
main()
{
int a=4;
int b=6;
Exchg3(a,b); //Exchg3时函数会将a,b 分别代替了x,y了,我们称x,y分别引用了a,b变量
Print(“a=%d,b=%d\n”, a, b);
}
x=6, y=4
a=6, b=2
最后对值传递与引用传递作一个比较:
1. 在函数定义格式上有不同:
值传递在定义处是:Exchg1(int x, int y);
引用传递在这义处是:Exchg1(int &x, int &y);
2. 调用时有相同的格式:
值传递:Exchg1(a,b);
引用传递:Exchg3(a,b);
3. 功能上是不同的:
值传递的函数里操作的不是a,b变量本身,只是将a,b值赋给了x,y函数里操作的只是x,y变量而不是a,b,显示a,b的值不会被Exchg1函数所修改。
引用传递Exchg3(a,b)函数里是用a,b分别代替了x,y。函数里操作的是a,b。