Monkey and Banana(最长上升子序列模型)笔记

链接:
Monkey and Banana

题目描述

这里一组研究人员正在设计一项实验,以测试猴子的智商。他们将香蕉挂在建筑物的屋顶,同时,提供一些砖块给这些猴子。如果猴子足够聪明,它应当能够通过合理的放置一些砖块建立一个塔,并爬上去吃他们最喜欢的香蕉。
研究人员有n种类型的砖块,每种类型的砖块都有无限个。第i块砖块的长宽高分别用xi,yi,zi来表示。 同时,由于砖块是可以旋转的,每个砖块的3条边可以组成6种不同的长宽高。
研究人员有n种类型的砖块,每种类型的砖块都有无限个。第i块砖块的长宽高分别用xi,yi,zi来表示。 同时,由于砖块是可以旋转的,每个砖块的3条边可以组成6种不同的长宽高。
你的任务是编写一个程序,计算猴子们最高可以堆出的砖块们的高度。

Input

输入文件包含多组测试数据。
每个测试用例的第一行包含一个整数n,代表不同种类的砖块数目。n<=30.
接下来n行,每行3个数,分别表示砖块的长宽高。
当n= 0的时候,无需输出任何答案,测试结束。

Output

对于每组测试数据,输出最大高度。格式:Case 第几组数据: maximum height = 最大高度;

Sample Input

1
10 20 30
2
6 8 10
5 5 5
7
1 1 1
2 2 2
3 3 3
4 4 4
5 5 5
6 6 6
7 7 7
5
31 41 59
26 53 58
97 93 23
84 62 64
33 83 27
0 

Sample Output

Case 1: maximum height = 40
Case 2: maximum height = 21
Case 3: maximum height = 28
Case 4: maximum height = 342 

题意理解

我们先看每个砖块的3条边可以组成6种不同的长宽高。,这里我们拿10,20,30举例,它可以构成(10,20,30),(10,30,20),(20,10,30),(20,30,10),(30,10,20),(30,20,10);这六种,我们可以规定长大于宽,这样就只有三种了,认为第一个是长,第二个是宽,第三个是高,(30,20,10),(30,10,20),(20,10,30),这三种;还有这里可以构成多种,是因为它说每种类型的砖块有无限个,实际上经过这种变形,每种砖块可以变成三种砖块;

解题思路

很多的题解都说这是最长上升子序列的变形,想一想这题,一开始感觉这有关系吗????,emmm,既然说是,那么我们比较一下,最长上升子序列是求最大子序列长度,我们这题是求最大高度,好像是有那么一点关系;
那么再看,最长上升子序列就是找几个数字,使其严格递增,其长度最长,而这题,是找几块砖,要求是上面的长和宽严格小于下面的长和宽,其实也就是某种角度的严格递减了属于,找出其可以堆出的最高的高度;这么一看,确实有点像;
然后我们想最长上升子序列是怎么求得,最长上升子序列是以第i个数字结尾,考虑其前一个数字是谁;而这题,就是在一堆砖中,那么就以第i块砖为最顶上的那一块,考虑其前一块是谁;行吧,貌似就是这样的;
注意:我们这规定了长和宽,所以是需要排序的,由于这题要用到结构体,可以利用sort排序,之前有总结关于sort排序的笔记;
代码:

#include<bits/stdc++.h>

using namespace std;

const int N=100;
//结构体,长宽高
struct rec
{
	int a,b,c;
};
rec q[N];//存n个长宽高
int f[N];//以第i块为最顶上的那一块的最大高度
int n;//每组砖的种类
int cnt=0;//一种砖可以变成三块,故以此作为统计
int T=0;//统计测试了几组,是为后面输出格式
//读入数据,放入q中
void putin(int a, int b, int c)
{
	cnt++;
	q[cnt].a=a;
	q[cnt].b=b;
	q[cnt].c=c;
}
//定义排序规则
bool cmp(rec A,rec B)
{
	if(A.a!=B.a)
		return A.a>B.a;
	else
		return A.b>B.b;	
}
int main()
{
	while(scanf("%d",&n)&&n)
	{
		T++;
		memset(q,0,sizeof(q));
		memset(f,0,sizeof(f));
		cnt=0;
		for(int i=1;i<=n;i++)
		{
			int t[3];
			scanf("%d%d%d",&t[0],&t[1],&t[2]);
			sort(t,t+3,greater<int>());
			putin(t[0],t[1],t[2]);
			putin(t[0],t[2],t[1]);
			putin(t[1],t[2],t[0]);
		}
		sort(q+1,q+cnt+1,cmp);
		//由最长上升子序列-->本题,以第i块为最顶上的一块,考虑前一块是哪个
		for(int i=1;i<=cnt;i++)
		{
			f[i]=q[i].c;//没有符合的,起码第i块是一定有的,即初始化
			for(int j=1;j<i;j++)
			{
			//上面的长和宽,分别严格小于下面的长和宽
				if(q[i].a<q[j].a&&q[i].b<q[j].b)
				{
					f[i]=max(f[i],f[j]+q[i].c);
				}
			}
		}
		//因为是以第i块为最顶上,所以要遍历整个数组,找出最大的哪一个,这个跟最长上升子序列一样的
		int res=0;
		for(int i=1;i<=cnt;i++)
			res=max(res,f[i]);
		printf("Case %d: maximum height = %d\n",T,res);
	}
	return 0;
} 

总结:
1,认真理解题意,分析是怎么来的;
2,完全我是不知道这题它是最长上升子序列,可能是题目做少了,又或是自己太菜看不出来,反正就是没看出来,太难联想了;
3,收获就是用到了sort排序,之前总结了一下,这次运用了一下,还挺好的,也算是复习了最长上升子序列吧;
4,慢慢来吧,慢慢啃呗,还能怎么样呢;

  • 5
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值