纪录动态规划(浙大1108FatMouse's Speed,杭电1069Monkey and Banana)解题报告

本文介绍了一种使用动态规划解决特定二维数组中最长序列问题的方法,并详细解释了如何记录回溯路径来输出序列中的元素。此外,还提供了一个具体实例,展示了如何处理涉及三维结构的类似问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

这一类型的题目很多,就是对一个二维数组进行动态规划,但题目所要求得符合要求的最长串一般都是可以不按输入的顺序来的,所以在进行动态规划之前要根据题目要求对数组进行排序。。。。

题目大意为,一个二维数组,每次输入w,s,要求

W[m[1]] < W[m[2]] < ... < W[m[n]]
and

S[m[1]] > S[m[2]] > ... > S[m[n]]
输出最长串的长度和串中元素数组下标。

因为要求原来输入数组的下标,所以本题用结构体保存比较方便,排序用qsort就可以。。

然后就是经典的邱最长串的问题了,经电解法:

for(i=0;i<n;i++){
   sum[i]=1;
}                 //sum[i],表示到t[I]的最长子传的长度。

for(i=1;i<n;i++){
   for(j=0;j<i;j++)
    if(t[j].a<t[i].a&&t[j].b>t[i].b&&sum[j]>=sum[i]){
     sum[i]++;
    }
   if(sum[i]>max){
    max=sum[i];
   }
}

max就是最长子串的长度。。。

接下来就是本题困扰了我很长时间的问题,怎么输出最长子串中的元素呢?

其实也是动态规划的做法。。即纪录每个元素,到这一元素的最长子串中,这一元素的上一元素的数组下表。将其保存在father[i]数组中(我的代码是用flag[i]表示的),flag[i]的初始值都设为-1,这样保证可以最后打出时能够走到最长子串的开头并及时停止,下面是动态规划的代码:

for(i=0;i<n;i++){
   sum[i]=1;
}
qsort(t,n,sizeof(t[0]),cmp);                 
max=1;
maxindex=0;
flag[0]=-1;
for(i=1;i<n;i++){
   flag[i]=-1;
   for(j=0;j<i;j++)
    if(t[j].a<t[i].a&&t[j].b>t[i].b&&sum[j]>=sum[i]){
     sum[i]++;
     flag[i]=j;            //不断纪录每一元素的符合要求的最长串的前一元素的坐标
    }
   if(sum[i]>max){
    max=sum[i];
    maxindex=i;
   }
}

 

最后打出时,就用反递推把最长子串的数组元素一个个打出就好了。

x = flag[maxindex];   
    while (x != -1){     //通过递推,把最长子串写入opp数组中
        opp[total] = x;   
        x =flag[x]; ++total;   
    }   
printf("%d/n",max);
for(i=total-1;i>=0;--i)
   printf("%d/n",t[opp[i]].num);
printf("%d/n",t[maxindex].num);

全部代码如下:

#include<stdio.h>
#include<stdlib.h>
typedef struct data{
int num;
int a;
int b;
};
struct data t[1001];
int flag[1001];
int cmp(const void *e1,const void *e2){
struct data *t1=(struct data *)e1;
struct data *t2=(struct data *)e2;
if(t1->a==t2->a) return t2->b-t1->b;
else return t1->a-t2->a;
}
void main(){
int i,n=0,j,max,p,maxindex,maxt,maxi,total=0,opp[1001];
int sum[1001];
int x;
while(scanf("%d%d",&t[n].a,&t[n].b)!=EOF){
   t[n].num=n+1;
   n++;

for(i=0;i<n;i++){
   sum[i]=1;
}
qsort(t,n,sizeof(t[0]),cmp);
max=1;
maxindex=0;
flag[0]=-1;
for(i=1;i<n;i++){
   flag[i]=-1;
   for(j=0;j<i;j++)
    if(t[j].a<t[i].a&&t[j].b>t[i].b&&sum[j]>=sum[i]){
     sum[i]++;
     flag[i]=j;            //不断纪录每一元素的符合要求的最长串的前一元素的坐标
    }
   if(sum[i]>max){
    max=sum[i];
    maxindex=i;
   }
}
x = flag[maxindex];   
    while (x != -1){     //通过递推,把最长子串写入opp数组中
        opp[total] = x;   
        x =flag[x]; ++total;   
    }   
printf("%d/n",max);
for(i=total-1;i>=0;--i)
   printf("%d/n",t[opp[i]].num);
printf("%d/n",t[maxindex].num);
}

同一类型的题目还有:http://acm.hdu.edu.cn/showproblem.php?pid=1069 monkey and banana,这一体看似很麻烦,其实就是把每种箱子的三种排放方式都写入一个数组,因为不可能有两种底完全一样的排放方式,所以这就构成了一个有三个元素的结构体数组,将其按照长宽高依次排序(注意排序前先要保证每种箱子的摆放方式必须保证长大于宽!!!!很重要,容易忽略,然后就是求高的最长子序列了,不同的是这个最长不是元素个数最长,而是高之和最大。。。。动态规划代码为:

for(i=0;i<=j;i++)
   sum[i]=t[i].g;                     //这里的sum[i]不再是1,而是箱子的高度!!
maxx=sum[0];
for(i=1;i<j;i++){
    max=0;
    for(p=0;p<i;p++)
        if((t[p].c>t[i].c&&t[p].k>t[i].k))           //||(t[p].k>t[p].c&&t[p].c>t[i].k)
     if(max<sum[p])
        max=sum[p];
    sum[i]=max+sum[i];
    if(sum[i]>maxx)
        maxx=sum[i];
}

最后打出max就是答案。。。本题代码为:

#include<stdio.h>
#include<stdlib.h>
typedef struct data{
int c,k,g;
};
struct data t[100];
int cmp(const void *e1,const void *e2){
struct data *t1=(struct data*)e1;
struct data *t2=(struct data*)e2;
if(t1->c!=t2->c) return t2->c-t1->c;
else{
   if(t1->k!=t2->k) return t2->k-t->k;
   else
    return t2->g-t1->g;
}
}
void getOther(struct data* block,int a,int b,int c){                
//如果长小于宽,就要转换!!

int t1;
block->c = a;
block->k= b;
block->g = c;    
if ((*block).c< (*block).k) {      
   t1=(*block).c;
   (*block).c=(*block).k;
   (*block).k=t1;
}
}
void main(){
int i,n,j,max,count=0,p,maxx,a,b,c;
int sum[100];
while(scanf("%d",&n)&&n!=0){
   count++;
   for(i=0,j=0;i<n;i++){
    scanf("%d%d%d",&a,&b,&c);
    getOther (&t[j++], a, b, c);
    getOther (&t[j++], c, b, a);
    getOther (&t[j++], a, c, b);
   }
qsort(t,j,sizeof(t[0]),cmp);               //对长宽高依次排序。
for(i=0;i<=j;i++)
   sum[i]=t[i].g;
maxx=sum[0];
for(i=1;i<j;i++){
   max=0;
   for(p=0;p<i;p++)
    if((t[p].c>t[i].c&&t[p].k>t[i].k))           //||(t[p].k>t[p].c&&t[p].c>t[i].k)
     if(max<sum[p])
      max=sum[p];
   sum[i]=max+sum[i];
   if(sum[i]>maxx)
    maxx=sum[i];
}
printf("Case %d: maximum height = %d/n",count,maxx);
}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值