分享

C语言数组当参数传递

 华灯初放l 2018-08-27

在学习C语言的过程中遇到数组作为参数传递的问题

一维数组:

  1. #include <stdio.h>
  2. int test2(int a[]){
  3. for(int i=0;i<5;i++){
  4. printf("%d",a[i]);
  5. }
  6. }
  7. int main(){
  8. int a[5] = {1,2,3,4,5},*p;
  9. p = a;
  10. test2(a);
  11. }


这样我们可以很顺利的在test去遍历这个数组a,当然我们还可能传递指针:

  1. int test1(int *p){
  2. for(int i=0;i<5;i++){
  3. printf("%d",p[i]);//我们在这里还可以用)*(p+i)来输出数组中的值
  4. }
  5. }
  6. int main(){
  7. int a[5] = {1,2,3,4,5},*p;
  8. p = a;
  9. test1(p);
  10. }


一般来数参数的传递是值传递,也就是说实参传给形参,形参发生改变时实参并不会改变,(单向)但是数组在传递的时候是地址传递,只要形参发生了变化,实参也会发生变化(双向)。

这样传递数组就会发现一个问题,我没有办法获取到数组的长度。获取数组的长度我们一般用:

sizeof(a)/sizeof(int)

  1. int test2(int a[]){
  2. int n = sizeof(a)/sizeof(int);
  3. for(int i=0;i<n;i++){
  4. printf("%d ",a[i]);
  5. a[i]++;
  6. }
  7. }
我们会发现n的值一直是2!为什么会这样呢!?
因为,a是函数参数,到了本函数中,a只是一个指针(地址,系统在本函数运行时,是不知道a所表示的地址有多大的数据存储空间,这里只是告诉函数:一个数据空间首地址),所以,sizoef(a)的结果是指针变量a占内存的大小,一般在64位机上是8个字节。int类型是4个字节,所以,结果永远是2,因此,我们要向获取数组长度要怎么办呢?

我可以在初始化数组的地方获取到数组的长度,作为参数传递过来:

  1. int test2(int a[],int n){
  2. for(int i=0;i<n;i++){
  3. printf("%d ",a[i]);
  4. a[i]++;
  5. }
  6. }
  7. int main(){
  8. int a[5] = {1,2,3,4,5},*p;
  9. int n = sizeof(a)/sizeof(int);
  10. test2(a,n);
  11. }

这样做我们可以获取到数组的长度。

二维数组:

二维数组作为参数传递是后我们不可以像以为数组那样直接,如:

  1. void test1(int a[][]){
  2. for(i = 0; i < 5; i++){
  3. for(j = 0; j < 5; j++){
  4. printf("%d ",a[i][j]);
  5. }
  6. }
  7. }
  8. int main(){
  9. int a[5][5],i,j;
  10. for(i = 0; i < 5; i++){
  11. for(j = 0; j < 5; j++){
  12. a[i][j] = i*5 + (j +1);
  13. }
  14. }
  15. test1(a);
  16. return 0;
  17. }

会发现编译都编译不通过,报“[Error] declaration of 'a' as multidimensional array must have bounds for all dimensions except the first”这个错,意思是多维数组的定义必须有一个除第一个之外的所有维度的边界,比如:

  1. void test1(int a[][5]){
  2. for(int i = 0; i < 5; i++){
  3. for(int j = 0; j < 5; j++){
  4. printf("%d ",a[i][j]);
  5. }
  6. }
  7. }
这样就OK了,但是我们是动态分配的数组不知道这个维度是多少的时候怎么办?这时候我们可以用指针当做一维数组来操作:
  1. void test1(int *p,int n){
  2. for(int i = 0; i < n; i++){
  3. printf("%d ",p[j]);
  4. }
  5. }
  6. int main(){
  7. int a[5][5],i,j;
  8. int *p;
  9. p = &a[0][0];
  10. for(i = 0; i < 5; i++){
  11. for(j = 0; j < 5; j++){
  12. a[i][j] = i*5 + (j +1);
  13. }
  14. }
  15. test1(p,25);
  16. return 0;
  17. }


这样我们发现不能更灵活的去定位到某一行某一列,这样我们需要手工改变寻址方式:
  1. void test2(int m,int n,int **p){//m,n是行和列,
  2. for(int i = 0; i < m; i++){
  3. for(int j = 0; j < n; j++){
  4. printf("%d ",*((int *)p+n*i+j));
  5. }
  6. }
  7. }
  8. int main(){
  9. int a[5][5],i,j;
  10. for(i = 0; i < 5; i++){
  11. for(j = 0; j < 5; j++){
  12. a[i][j] = i*5 + (j +1);
  13. }
  14. }
  15. test2(5,5,(int **)a);
  16. return 0;
  17. }







    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多