FatMouse's Speed

题目

样例

这道题目就是LIS加上结构体来记录信息,从而做到老鼠越胖速度越慢的情况!

#include <iostream>
#include <cstdlib>
#include <algorithm>
#include <cmath>
#include <map>
#include <set>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#define mem(a,b) memset(a,b,sizeof(a))
#define INF 0x3f3f3f3f
#define _for(i,a,b) for( int i=(a); i<(b); ++i)
#define _rep(i,a,b) for( int i=(a); i<=(b); ++i)
typedef long long ll;
using namespace std;
const int MAX = 10010;//题目中给了最多1000只老鼠但是没说给几组测试数据所以尽量开大点,这里输出时是特殊输出:Ctrl+Z
int DP[MAX];//最大子序列
int rem[MAX];//记录符合条件的老鼠
using namespace std;
typedef struct Mouse
{
	int weight;//重量
	int speed;//速度
	int num;//编号
}mice;
mice s[MAX];
bool cmp(mice x, mice y)//这里安体重从大到小,速度从小到大排序当然也可以全部反过来
{
	if (x.weight == y.weight)
		return x.speed < y.speed;
	else
		return x.weight > y.weight;
}
int main()
{
	int n, m, maxn, i, j, k, t, mark;
	for (i = 1; i < MAX; i++)//初始化,每只记录的老鼠都是自己
		rem[i] = i;
	k = 1;
	while (scanf("%d%d", &s[k].weight, &s[k].speed) != EOF)//特殊输入输出,输入完数据按下Ctrl+Z就会出数据
	{
		s[k].num = k;
		k += 1;
	}
	sort(s + 1, s + k, cmp);//排序
	mem(DP,0);//初始化为0
	for (i = 1, m = 0; i < k; i++)
	{
		maxn = 0;//maxn表示最大的子序列长度即找到符合条件的最多的老鼠数量
		for (j = 1; j < i; j++)
		{
			if (s[i].weight<s[j].weight && s[i].speed>s[j].speed)//如果符合条件
			{
				if (maxn < DP[j])//如果此时数量大于记录的最大数量
				{
					maxn = DP[j];//把此时数量赋给maxn
					mark = s[j].num;//并且记录此时这只老鼠的编号
				}
			}
			if (maxn)//如果最大数量存在
				rem[s[i].num] = mark;//把用于比较老鼠编号和记录的老鼠编号对应记录下来
			DP[i] = maxn + 1;//如果不存在此时maxn==0加上1就是1即就算找不到也有它自己啊--找到了那就更新原来的值即+1
			if (m < DP[i])//m是记录符合条件的最大老鼠数量
			{
				m = DP[i];
				t = s[i].num;//t是记录最后一只符合条件的老鼠编号方便输出
			}
		}
	}
	if (m == 1)//当m等于1时即没有符合条件的其他老鼠那只能是随便哪只都可以输出所以我就固定输出1(数量)和1(编号)
	{
		cout << 1 << endl << 1 << endl;
	}
	else//否则
	{
		cout << m << endl;//输出符合条件的最大数量
		while (rem[t] != t)//如果此时老鼠编号对应的老鼠不是自己
		{
			cout << t << endl;//那么输出这只老鼠的编号
			t = rem[t];//并且把对应的老鼠的编号重新赋给t
		}
		cout << t << endl;//因为对应的最后一只老鼠的编号的rem[对应的最后一只老鼠]=它自己。。所以我们要把对应的最后一只老鼠的编号单独输出来
	}
	return 0;//完毕
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值