我在接受采访时被问到这个问题。我得到一个随机数的二维数组(数字可以重复),我需要对它们进行行和列排序,即所有的行和列都应该排序。任何人都可以解释如何有效地做到这一点(最小的时间和空间复杂性)。如果您可以使用C或C ++提供代码,那么这将非常有用
答案 0 :(得分:2)
我的想法是,二维数组可以(在某些情况下)被视为一维数组,因为它的存储器存储。如果没有,您可以将其复制到写入自定义排序函数的一维数组中,该函数使用将索引从1D转换为2D的函数。
这是一个使用qsort函数的例子:
#include <stdio.h>
#include <stdlib.h>
int matrix[4][4];
void print_matrix() {
int i, j;
for (i = 0; i < 4; i++) {
for (j = 0; j < 4; j++) {
printf("%d ", matrix[i][j]);
}
printf("\n");
}
}
int compar(const void *a, const void *b) {
int ia = *((int*)a);
int ib = *((int*)b);
return ia - ib;
}
int main() {
int i, j;
// Init of a 2D array with random numbers:
for (i = 0; i < 4; i++) {
for (j = 0; j < 4; j++) {
matrix[i][j] = random() % 10;
}
}
// Before:
printf("Before:\n");
print_matrix();
// This array can be considered as a big 1D array.
qsort(matrix, 16, sizeof(int), compar);
// After:
printf("After:\n");
print_matrix();
return 0;
}
输出:
Before:
3 6 7 5
3 5 6 2
9 1 2 7
0 9 3 6
After:
0 1 2 2
3 3 3 5
5 6 6 6
7 7 9 9
编辑: OP让我避免使用qsort ...所以这里有一个快速排序能够对2D数组进行排序:
#include <stdio.h>
#include <stdlib.h>
void print_matrix(int **matrix, int rows, int cols) {
int i, j;
for (i = 0; i < rows; i++) {
for (j = 0; j < cols; j++) {
printf("%d ", matrix[i][j]);
}
printf("\n");
}
}
void swap(int *a, int *b) {
int buf = *a;
*a = *b;
*b = buf;
}
int partition(int **a, int l, int r, int c) {
int i;
// Left pivot
int pivot_val = a[l/c][l%c];
// Move pivot to end
swap(&a[l/c][l%c], &a[r/c][r%c]);
// If <= to the pivot value, swap
int j = l;
for (i = l; i < r; i++) {
if (a[i/c][i%c] <= pivot_val) {
swap(&a[i/c][i%c], &a[j/c][j%c]);
j++;
}
}
// Move pivot to its place.
swap(&a[j/c][j%c], &a[r/c][r%c]);
return j;
}
void quicksort_r(int **a, int l, int r, int c) {
if (l < r) {
int pivot = partition(a, l, r, c);
quicksort_r(a, l, pivot-1, c);
quicksort_r(a, pivot+1, r, c);
}
}
void quicksort(int **a, int rows, int cols) {
quicksort_r(a, 0, rows * cols - 1, cols);
}
int main() {
int i, j;
int rows = 5;
int cols = 4;
int **matrix = malloc(sizeof(int*) * rows);
// Init of a 2D array with random numbers:
for (i = 0; i < rows; i++) {
matrix[i] = malloc(sizeof(int) * cols);
for (j = 0; j < cols; j++) {
matrix[i][j] = random() % 10;
}
}
// Before:
printf("Before:\n");
print_matrix(matrix, rows, cols);
quicksort(matrix, rows, cols);
// After:
printf("After:\n");
print_matrix(matrix, rows, cols);
return 0;
}
给出了:
Before:
3 6 7 5
3 5 6 2
9 1 2 7
0 9 3 6
0 6 2 6
After:
0 0 1 2
2 2 3 3
3 5 5 6
6 6 6 6
7 7 9 9
Edit2:之后我意识到方形矩阵还有另一个明显的解决方案:
我们来看第一个例子:
0 1 2 2
3 3 3 5
5 6 6 6
7 7 9 9
还有:
0 3 5 7
1 3 6 7
2 3 6 9
2 5 6 9
但是对于第二个例子:
0 0 1 2
2 2 3 3
3 5 5 6
6 6 6 6
7 7 9 9
和
0 2 3 6
0 2 5 6
1 3 5 6
2 3 6 6
7 7 9 9
这意味着我们可能会做一个能够提供所有解决方案的专门算法或一个试图最小化移动次数的算法,我不知道。事实上,这是一个非常有趣的问题。
答案 1 :(得分:0)
尝试使用并改编它......
void sort(int MyArray[8][8])
{
for(int ir=0;ir<8;ir++)
{
for(int ic=0;ic<8;ic++)
{
for(int jr=0;jr<8;jr++)
{
for(int jc=0;jc<8;jc++)
{
if(MyArray[ir][jr]>MyArray[ic][jc])
{
int temp=MyArray[ir][jr];
MyArray[ir][jr]=MyArray[ic][jc];
MyArray[ic][jc]=temp;
}
}
}
}
}
}
这是一个简单的冒泡排序。它有效地运行在O(n ^ 4)上你可以通过编写诸如quicksort之类的算法来提高效率。我从未见过甚至想过在2D阵列上尝试快速排序。这并不容易。您可以尝试将所有内容添加到单个阵列中,快速排序并将所有内容复制回2D阵列。但是数组必须非常大才能更有效...如果有人在这里为2D数组提供快速排序或合并排序算法。他应该获得一枚奖章!这是硬核