解题思路:
1.假设二维数组为  N=3

0   1   2    H=0  代表行:第一行 

3   4   5    H=1

6   7   8    H=2

H=3,没有这行,结束;

注意:以下程序使用了一维数组代替二维数组:(因为刚开始用全排列做的,一维数组直接把全排列搬过来就可,之后换方法就没改)

每一行的第一个元素,在一维数组中的下标为[H*N];

每行的最后一个元素,在一位数组中的下表为[H*N+N-1],即:[(H+1)*N-1]

 以下是递归关系:(得到所有不同矩阵)

(1).到第三行(最后一行)右移N(N=3)次,每次得到一个新的矩阵,每得到一个矩阵,求其最大值;

(2).到第二行移动三次,每移动一次,做一遍(1);

(3).到第一行移动三次,每移动一次,做一遍(2);

 

  优化:

  因为

1   0   2     和  0   1   2

4   3   5           3   4   5

7   6   8           6   7   8      .....是一样的

所以第一行可以不用右移动;若不能理解的话,把所有的写出来,看看;

递归关系简化为:

(1).到第三行(最后一行)右移N(N=3)次,每次得到一个新的矩阵,每得到一个矩阵,求其最大值;

(2).到第二行移动三次,每移动一次,做一遍(1);


注意,简化后,因为第一行没有排序,当输入N=1的时候,不能得出结果(具体看函数),所以当N==1时,结果单独输出;

if(N==1)//N==1直接输出结果,原因向下看
   printf("%d\n",A[0]);
          else         //否则进行排列递归,从第二行开始
          {
            pailie(A,N,1); //1表示从第二行开始移动(因为优化第一行不移动了),当N等于1时,看下面pailie
                               //实现,H==N,直接就返回了,所以N==1在上面单独输出
            free(A);
            printf("%d\n",comax_min);
          }

递归实现;

void pailie(int *A,int N,int H)
{

           if(H==N)     //当H==N,看上面例子,可以知道,前面已经到最后一行,所以返回上一个函数
             return;
            for(int j=0;j<N;j++)    //每一行移动N次
              {

                       pailie(A,N,H+1);   //递归到下一行

                      int t=A[N*(H+1)-1];     //记录这一行的最后一个元素
                       for(int i=N*(H+1)-1;i>N*H;i--)  //移动
                            A[i]=A[i-1];     
                              A[N*H]=t;

                        Colmax(A,N);   //求矩阵的最大值,同时里面会记录所有矩阵最大值的最小一个
              }
return ;
}


参考代码:

#include<stdio.h>
#include<malloc.h>
int comax_min;
void pailie(int *A,int N,int H);
void input(int *A,int N);
void Colmax(int *A,int N);
int main()
{
   int *A;
   int N;

   while(scanf("%d",&N)&&N!=-1)
   {
                 comax_min=100000;
                 A=(int *)malloc((N*N)*sizeof(int));
                 input(A,N*N);

                  if(N==1)
                     printf("%d\n",A[0]);
                   else
                       {
                           pailie(A,N,1);
                           free(A);
                           printf("%d\n",comax_min);
                       }

   }
return 0;
}

/*===================================================*/
void input(int *A,int N)
{
   for(int i=0;i<N;i++)
    scanf("%d",&A[i]);
}
/*===================================================*/
void pailie(int *A,int N,int H)
{

           if(H==N)
             return;
            for(int j=0;j<N;j++)
              {

                pailie(A,N,H+1);

                      int t=A[N*(H+1)-1];
                       for(int i=N*(H+1)-1;i>N*H;i--)
                            A[i]=A[i-1];
                              A[N*H]=t;

                        Colmax(A,N);

              }


return ;
}

/*===================================================*/
void Colmax(int *A,int N)   //求矩阵的最大值,并且求出所有矩阵中最大值最小的一个
{

   int max,sum=0;

   for(int j=0;j<N;j++)  //先求第一例的和
     sum+= A[j*N];
     max=sum;  //让最大值先等于第一列的和
     
   for(int i=1;i<N;i++)    //再求后面的列的和
   {   sum=0;
      for(int k=0;k<N;k++)
        {
          sum+=A[k*N+i];
        }
     if(max<sum)  //后面列的和更大,则更行max
     max=sum;

   }
  if(comax_min>max)   //记录所有矩阵最大值中最小的一个
    comax_min=max;
}


下面介绍全排列法以上方法就是它来的:(此方法仅供观赏,测试电脑性能)

同样把除了第一行以外的行,做全排列;

0   1   2    H=0  代表行:第一行 

3   4   5    H=1

6   7   8    H=2


把第二行做全排列,每得到一种排列,转向排列它的下一行,(同样每得到一种排列,排列它的下一行);若是最后一行,排列这一行,每一种排列求出这个矩阵的最大值。

//整个实现过程就是全排列里面,加上递归全排列,和求矩阵最大值
void pailie(int *A,int index,int length,int H,int N)
{

   int j=0;

   if(index==length)  //得到一种排列
    {
        if(H==N-1)   //看是不是最后一行
           Colmax(A,N);   //是最后一行,求矩阵最大值
          
           else
             pailie(A,(H+1)*N,(H+2)*N,H+1,N);  //不是最后一行,递归下一行

    }
    else
     for(j = index;j < length; j++)
        {
            swap(&A[j],&A[index]);
            pailie(A,index+1,length,H,N);
            swap(&A[j],&A[index]);
        }
return ;
}


参考代码:

#include<stdio.h>
#include<malloc.h>
int comax_min=0;
void pailie(int *A,int index,int length,int H,int N);
void swap(int *x,int *y);
void input(int *A,int N);
void Colmax(int *A,int N);
int main()
{
   int *A;
   int N;
   while(scanf("%d",&N)&&N!=-1)
   {
         comax_min=100000;
        A=(int *)malloc((N*N)*sizeof(int));
        if(N==1)
         {
         input(A,N*N);
         printf("%d\n",A[0]);
         }
         else
           {
               input(A,N*N);
               pailie(A,1*N,N+N,1,N);
               printf("%d\n",comax_min);
           }
   }
return 0;
}

/*===================================================*/
void input(int *A,int N)
{
   for(int i=0;i<N;i++)
    scanf("%d",&A[i]);
}
/*===================================================*/
void pailie(int *A,int index,int length,int H,int N)
{

   int j=0;

   if(index==length)
    {
        if(H==N-1)
           Colmax(A,N);

           else
             pailie(A,(H+1)*N,(H+2)*N,H+1,N);

    }
    else
     for(j = index;j < length; j++)
        {
            swap(&A[j],&A[index]);
            pailie(A,index+1,length,H,N);
            swap(&A[j],&A[index]);
        }
return ;
}

/*===================================================*/
void Colmax(int *A,int N)
{

   int max,sum=0;

   for(int j=0;j<N;j++)
     sum+= A[j*N];
     max=sum;
   for(int i=1;i<N;i++)
   {   sum=0;
      for(int k=0;k<N;k++)
        {
          sum+=A[k*N+i];
        }
     if(max<sum)
     max=sum;
   }

  if(comax_min>max)
    comax_min=max;
}
/*===================================================*/
void swap(int *x,int *y)
{
  int z=(*x);
  (*x)=(*y);
  (*y)=z;
return ;
}

别忘点赞哦-.-

点赞(21)
 

0.0分

6 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 5 条评论

【‘hang行】、’ 3年前 回复TA
好家伙  nb
陶仁勇 6年前 回复TA
太溜了吧
lalalala 6年前 回复TA
@lalalala 改改还是可行的!
Manchester 6年前 回复TA
@lalalala 衷心提示,仅供观赏,测试电脑性能
lalalala 6年前 回复TA
第二个容易超时