如何在C中迭代遍历动态的矩形矩阵?

huangapple go评论85阅读模式
英文:

How to iterate through a dynamic, rectangular matrix in C?

问题

Sure, here's the translated code portion you requested:

  1. 我需要在C语言中使用动态内存中的指针创建一个矩阵,填充它们以随机数,然后打印出来。
  2. 这是大学作业的一部分(我必须为矩阵编写一整套函数),但是就帖子的目的而言,我认为我成功地找到并隔离了有问题的部分。以下是代码:
  3. ```c
  4. #include <stdio.h>
  5. #include <stdlib.h>
  6. #include <time.h>
  7. #include <stdbool.h>
  8. int main()
  9. {
  10. int rows = 2;
  11. int cols = 3;
  12. int **matrix = malloc(sizeof(int*) * rows); //声明和分配矩阵的动态内存
  13. for(int i=0;i<rows;i++)
  14. *(matrix+i) = malloc(sizeof(int) * cols);
  15. //生成矩阵的随机数:
  16. srand(time(NULL));
  17. for(int i=0;i<rows;i++)
  18. for(int j=0;j<cols;j++)
  19. *(*(matrix+j)+i) = rand() % 20;
  20. //打印矩阵:
  21. printf("{\t");
  22. for(int i=0;i<rows;i++)
  23. {
  24. int j;
  25. if(i>0)
  26. printf("\t");
  27. printf("{\t");
  28. for(j=0;j<cols;j++)
  29. printf("%d\t",*(*(matrix + j) + i));
  30. printf("}");
  31. if(i<rows-1)
  32. printf("\n");
  33. }
  34. printf("\t}\n\n");
  35. //销毁矩阵
  36. for(int i=0;i<rows;i++)
  37. free(matrix[i]);
  38. free(matrix);
  39. matrix = NULL;
  40. return 0;
  41. }

调试器在这里停止:

*(*(matrix+j)+i) = rand() % 20;

显然,在尝试为第二行的第一列生成随机数时(即首次更改行时),调试器无法访问该位置的内存,然后程序崩溃。

我尝试更改行数和列数,发现只有在行数小于列数时才会崩溃,这很奇怪,因为当它们相等或更多时,它运行得很好。

我正在尝试使用2行和3列。期望的输出是:

  1. { { 5 17 3 }
  2. { 1 8 11 } }

实际发生的是:

Process returned -1073741819 (0xC0000005)

当调试器停止(生成随机数时)时,变量的值如下:

  1. i = 0
  2. j = 2
  3. **(matrix + 2)(即 matrix[0][2]= 无法访问地址为 0xabababababababab 的内存

起初,我怀疑在为矩阵分配内存时出现了问题,但正如我所说,当矩阵是正方形时,它运行得很好,所以我真的不知道问题出在哪里。我知道StackOverflow社区不喜欢有人发布“我不知道我做错了什么,请帮助”的问题,但我确实花了很长时间尝试,无法找到问题的根本原因。请有人能够为我解答一下?

  1. <details>
  2. <summary>英文:</summary>
  3. I have to create a matrix with pointers in dynamic memory in C, fill it with random numbers then print it.
  4. This is part of a larger assignment from college (I had to do a whole library of functions for matrixes) but for the purposes of the post, I think I managed to track down and isolate the problematic bits. Here&#39;s the code:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <stdbool.h>

int main()
{
int rows = 2;
int cols = 3;
int *matrix = malloc(sizeof(int) * rows); //declaring and allocating dynamic memory for the matrix

  1. for(int i=0;i&lt;rows;i++)
  2. *(matrix+i) = malloc(sizeof(int) * cols);
  3. //generating random numbers for the matrix:
  4. srand(time(NULL));
  5. for(int i=0;i&lt;rows;i++)
  6. for(int j=0;j&lt;cols;j++)
  7. *(*(matrix+j)+i) = rand() % 20;
  8. //printing matrix:
  9. printf(&quot;{\t&quot;);
  10. for(int i=0;i&lt;rows;i++)
  11. {
  12. int j;
  13. if(i&gt;0)
  14. printf(&quot;\t&quot;);
  15. printf(&quot;{\t&quot;);
  16. for(j=0;j&lt;cols;j++)
  17. printf(&quot;%d\t&quot;,*(*(matrix + j) + i));
  18. printf(&quot;}&quot;);
  19. if(i&lt;rows-1)
  20. printf(&quot;\n&quot;);
  21. }
  22. printf(&quot;\t}\n\n&quot;);
  23. //destroying the matrix
  24. for(int i=0;i&lt;rows;i++)
  25. free(matrix[i]);
  26. free(matrix);
  27. matrix = NULL;
  28. return 0;

}

  1. The debugger stops here:
  2. `*(*(matrix+j)+i) = rand() % 20;`
  3. Apparently, when trying to generate a random number for the first column of the second row (i.e. when it first changes rows) the debugger &quot;cannot access memory&quot; for that position, then the program crashes.
  4. I tried changing the amount of rows and columns and I found it only crashes when row size is less than column size, which is weird because when it&#39;s equal or more, it works just fine.
  5. I&#39;m trying with 2 rows by 3 columns. Expectation:

{ { 5 17 3 }
{ 1 8 11 } }

  1. What actually happens:
  2. `Process returned -1073741819 (0xC0000005)`
  3. Values of the variables when the debugger stops (at generating random numbers):

i = 0
j = 2
**(matrix + 2) (aka matrix[0][2]) = Cannot access memory at address 0xabababababababab

  1. At first I suspected I messed up when allocating memory for the matrix, but as I said when the matrix is square it does work fine, so I really don&#39;t know what&#39;s the issue here. I know the stackOverflow community doesn&#39;t like when someone posts &quot;I don&#39;t know what I did wrong please help&quot; kind of thing but I&#39;ve really been a long time trying and couldn&#39;t get to the root of the problem.
  2. Can someone please shed some light on this for me?
  3. </details>
  4. # 答案1
  5. **得分**: 2
  6. These nested for loops contain a bug
  7. 这些嵌套的for循环包含一个错误
  8. ```c
  9. for(int i=0;i<rows;i++)
  10. for(int j=0;j<cols;j++)
  11. *(*(matrix+i)+j) = rand() % 20;

这个指针matrix指向一个指针数组,这些指针又指向"rows"。所以表达式matrix + i指向第i行。然而,你却使用了索引j

也就是说,你需要写成

  1. *(*(matrix+i)+j) = rand() % 20;

而不是

  1. *(*(matrix+j)+i) = rand() % 20;

否则,当cols的值大于rows的值时,表达式matrix + j可能会访问分配的指针数组(行)之外的内存。

在printf的这个调用中也存在同样的问题

  1. printf("%d\t",*(*(matrix + j) + i));

你需要写成

  1. printf("%d\t",*(*(matrix + i) + j));

也就是说,你实际上需要访问一个元素matrix[i][j]。这个表达式可以重写为

  1. ( *( matrix + i ) )[j]

然后再重写为

  1. *( *( matrix + i ) + j )

或者以另一种方式重写为

  1. *( matrix[i] + j )

然后

  1. *( *( matrix + i ) + j )

为了引起你对下标操作符的兴趣,研究一下这些等价表达式,用来访问一个声明为

  1. T a[M][N];

的二维数组的元素。这些表达式分别是

  1. a[i][j]
  2. i[a][j]
  3. j[a[i]]
  4. j[i[a]]
  5. *( *( a + i ) + j )
  6. *( a[i] + j )
  7. *( i[a] + j )
  8. ( *( a + i ) )[j]
  9. j[*( a + i ) ]
英文:

These nested for loops contain a bug

  1. for(int i=0;i&lt;rows;i++)
  2. for(int j=0;j&lt;cols;j++)
  3. *(*(matrix+j)+i) = rand() % 20;

The pointer matrix points to an array of pointers that in turn point to "rows". So the expression matrix + i points to the i-th "row". However instead you are using the index j.

That is you need to write

  1. *(*(matrix+i)+j) = rand() % 20;

instead of

  1. *(*(matrix+j)+i) = rand() % 20;

Otherwise when the value of cols is greater than the value of rows the expression matrix + j can access memory outside the allocated array of pointers (rows).

The same problem exists in this call of printf

  1. printf(&quot;%d\t&quot;,*(*(matrix + j) + i));

where you have to write

  1. printf(&quot;%d\t&quot;,*(*(matrix + i) + j));

That is you need actually to access an element matrix[i][j]. This expression may be rewritten like

  1. ( *( matrix + i ) )[j]

that in turn may be rewritten like

  1. *( *( matrix + i ) + j )

Or in other way it may be rewritten like

  1. *( matrix[i] + j )

and then

  1. *( *( matrix + i ) + j )

To raise your interest to the subscript operator investigate these equivalent expressions to access elements of a two-dimensional array declared for example like

  1. T a[M][N];

The expressions are

  1. a[i][j]
  2. i[a][j]
  3. j[a[i]]
  4. j[i[a]]
  5. *( *( a + i ) + j )
  6. *( a[i] + j )
  7. *( i[a] + j )
  8. ( *( a + i ) )[j]
  9. j[*( a + i ) ]

huangapple
  • 本文由 发表于 2023年4月19日 16:20:23
  • 转载请务必保留本文链接:https://go.coder-hub.com/76052217.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定