洛谷P1311题解

先上思路:用结构体存每个客栈的色调和最低消费;用s[i][j]存i色调第j个客栈的编号,然后遍历s中的各种相同色调的客栈来找合法的方案;w[i]存储到第i个客栈为止满足最低消费的客栈数量,从而通过判断每两个客栈区间的w值有无变化而判断这两个客栈是不是一种合法方案。

这道题的难度就是对各种细节的考虑周不周到以及怎么处理存储这些输入的数据(当用了vector存各个色调信息之后不能说是完全不难吧,只能说是非常轻松^_^)

代码中除了基本的暴力递推之外就是用了w数组来便于判断某两个客栈之间有没有满足最低消费的客栈,从而方便判断是否为一种合理选择;然后就是在递推过程中用了一个简单的优化来减少代码的时间复杂度(虽然不知道直接暴力不加优化会不会tle,原谅我懒得试一下了qaq)

总的来说实在是没有啥难度,要说感想吧,只能说STL大法好hhh


(各种细节还是一如既往的在注释中写得很清楚嗷!!!)

Code:

//思路:用结构体存每个客栈的色调和最低消费,用s[i][j]存i色调第j个客栈的编号,w[i]存到第i个客栈为止满足
//最低消费的客栈数量
#define _CRT_SECURE_NO_WARNINGS 1
#define MAXN 200005
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
using namespace std;
int n, p, k;
struct kezhan
{
	int a;//色调
	int b;//最低消费
}c[MAXN];
int w[MAXN];
long long ans;
vector<int>s[55];//用vector存色调信息
int main(){
	cin >> n >> k >> p;
	for (int i = 1; i <= n; i++)
	{
		cin >> c[i].a >> c[i].b;
		s[c[i].a].push_back(i);//将客栈i存入该色调的vector中
		if (c[i].b <= p)//若可以接受消费
		{
			w[i] = w[i - 1] + 1;//从前一个位置的总数基础上再+1,为到第i个位置满足条件的客栈总数
		}
		else//这个点不满足消费要求
		{
			w[i] = w[i - 1];
		}
	}
	for (int i = 0; i < k; i++)
	{
		for (int j = 0; j < s[i].size(); j++)//遍历这个色调的所有客栈
		{
			for (int l = j + 1; l < s[i].size(); l++)
			{
				if (j != l && w[s[i][l]]!=w[s[i][j]-1])//说明s[i][j]和s[i][l]之间至少一个客栈满足要求
				{
					ans += s[i].size() - l;//优化剪枝,若s[i][j]的客栈之后只要有一个能满足,则其后的不需要再判断,一定满足
					break;//直接下一个j
				}
			}
		}
	}
	cout << ans;
	return 0;
} 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值