在C语言学习过程中出现的问题,使得自己对二级指针有一个更深的了解。
原问题如下:给一个不多于5位的正整数,要求:一、求它是几位数,二、逆序打印出各位数字。针对上问题,原本想写一个函数,返回值为位数,准备传入指针变量使得传出逆序后各位数字的数组,结果问题出现了,原始代码如下:
//No.29 给一个不多于5位的正整数,要求:一、求它是几位数,二、逆序打印出各位数字
int weishu(int n, int* print)
{
int tag = 0, temp = n;
while (temp > 0)
{
temp = temp / 10;
tag++;
}
print = (int*)malloc(sizeof(int) * tag);
for (int i = 0; i < tag; i++)
{
print[i] = n % 10;
n = n / 10;
}
return tag;
}
int main()
{
int* num = NULL;
int i = weishu(567890,num);
for (int j = 0; j < i; j++)
{
printf("%d\t", num[j]);
}
printf("\n");
}
运行后,出现错误:
到这里,知道自己是定义的num位置指针使用错误,导致的内存访问冲突,由于指针学习的不好,所以具体原因不清楚。经过各方资料查找,发现是定义的函数中指针使用出现问题。
问题出现前我想的:通过函数的返回值可以返回可以得到具体整数的位数,通过一个指针变量获得逆序后的数组。由于指针记录地址,所以认为指针可以带回主函数相应的数组,想的很好,但程序报错!
实际问题:问题出现后,经过多方查找,发现原因竟是函数的值传递!!可是形参使用指针,明明是地址,为何值传递??以下进行分析:
调用weishu函数时,相当于
int weishu(int n=567890;int *print=num)
而num是NULL,这里等于对print指针进行一个定义而已,实际情况下,并没有将主函数中num的地址传入函数weishu(),所以最终函数也不可能将数组返回给num,所以num一直都是NULL,所以一直报内存错误。再接着刚刚的print指针分析,后面确实让print指向开辟的内存,内存中也确实存储了逆序排列后的数字,但是weishu()函数执行完成后,该内存指针print相当于丢失了,并没有返回给num。
所以,想要在函数weishu()中将逆序数组返回给num,必须要传入num的地址,而又因为num本身就是指针变量,所以weishu()形参必须是指向num地址的二级指针。改进后的程序如下,可以正常运行:
//No.29 给一个不多于5位的正整数,要求:一、求它是几位数,二、逆序打印出各位数字
int weishu(int n, int** print) //返回的是位数
{
int tag = 0, temp = n;
while (temp > 0)
{
temp = temp / 10;
tag++;
}
*print = (int*)malloc(sizeof(int) * tag);
for (int i = 0; i < tag; i++)
{
(*print)[i] = n % 10;
n = n / 10;
}
return tag;
}
int main()
{
int* num = NULL;
int i = weishu(567890,&num);
for (int j = 0; j < i; j++)
{
printf("%d\t", num[j]);
}
printf("\n");
}
小结:不要因为指针是指针变量就想着直接传入即可,实际情况下,指针也不过是一个变量,只是其内容为一个地址,如果想要让它是一个实参并带回一个地址,还是需要使用变量的地址,即指针的地址、二级指针!!!