CCF刷题计划——垦田计划(手握map砍竹子)

垦田计划

计算机软件能力认证考试系统

刷题最有用的一集!做这道题的时候,我深刻感受到之前刷的题是有效果的QWQ。如果是以前,优先队列我是想都不会想的,如果是中间,优先队列只得70我就放弃了,但是现在,我刷过蓝桥杯 砍竹子,也就对这道题的优化有了眉目。

本题可以学到的:自定义类型优先队列的使用、map实现横向"砍竹子"

使用优先队列还超时了,这是我没想到的,但是好歹得了70分,而且复习了优先队列,还是不亏的。经过我的优化,还是把这道题做出来了。

70分:

cpp 复制代码
#include <iostream>
#include <queue>
using namespace std;
#define ll long long 
const int N=1e5+2; 
//思路:贪心吧,先将最高的那个打下来,放回去,然后再找最高,再打,直到资源用完或者最高的那个无法再减。
//看的这类问题就想起了 优先队列 

struct Infor	//田地信息 
{
	int time,cost;	//时间,减少一天的花费
	
	//使用自定义类型的优先队列需要重载运算符,多写几次就记住了
	Infor(int t,int c)
	{
		time=t;	cost=c;	
	} 
	bool operator < (const Infor& a) const	//大顶堆 
	{
		return time<a.time;
	}
};

priority_queue<Infor>pq;	// <当前开垦天数,减少一天的花费> 
int n,m,k;	//区域总数、资源总量、每块最少开垦天数 


int main()
{
	cin>>n>>m>>k;
	int time,cost;
	for(int i=0;i<n;i++)
	{
		cin>>time>>cost;
		pq.push({time,cost});
	} 
	
	while(m)	//当资源没用完 
	{
		time=pq.top().time;
		cost=pq.top().cost;
		
		if(time==k || m-cost<0)	//如果当前最大天数已经无法再减少
			break; 
		
		//到这都是能减少的
		m-= cost;
		time--;
		
		pq.pop();	//删除旧的 
		pq.push({time,cost});			//添加新的 
	}
	cout<<pq.top().time;
	return 0;
}

**优化的思绪(怎么想到这样优化的):**仔细想想还有哪里可以优化呢?其实break的条件可以再优化一下,想想,如果有100个田,有99块都能被材料减少,但是就是有一个无法满足,导致这些田都白减了,会浪费很多时间。如果可以有一个更高效的判断,或者不一个一个来减,直接减一截,那是不是好很多。

对!一次性减一截!这里我想起来之前刷蓝桥杯那个砍竹子,我们不一定要将一个田减一截,而可以将相同天数的田全部都减1!前者是纵向砍一大段,后者是横向砍一片,但是这一片都只向下减少了1。

经过我漫长的优化......(我一开始用map还没做出来,因为将totalcost的计算和向下迭代复杂化了,下面的代码就很满意)

AC:

cpp 复制代码
#include<iostream>
#include <map>
#include <vector>
using namespace std;
#define ll long long 
const int N=1e5+2;
const int inf=0x3f3f3f3f;

int n,m,k;
map<int,vector<int>>mp;	// <天数,相同天数的id集合> 
vector<int>c(N);
int ans=inf;

int main()
{
	cin>>n>>m>>k;
	int time,cost; 
	for(int i=0;i<n;i++)
	{
		cin>>time>>c[i];
		mp[time].push_back(i);
	}
	
	ll totalCost=0;
	
	//mp也是有顺序的,我们可以反向遍历将最大的key(也就是这里的time)进行处理
	auto it=mp.end();	//先指向最大的时间
	it--; 
	time=it->first;
	vector<int>cur=it->second;
	
	while(m>0)
	{
		for(auto id:cur)	//先将cur指向的田地加入到Totalcost   
			totalCost+=c[id];	//因为每次其实都是在上次的基础上进行负重增加,所以没必要从头到尾计算 
		
		if(m<totalCost || time==k)
		{
			cout<<time;
			return 0;
		}
		m-=totalCost;
		
		//逐渐向低天靠拢 
		time--;
		cur=mp[time];
	}
	return 0;
}
相关推荐
程序员老舅几秒前
【无标题】
c++·嵌入式·八股文·c++八股文·八股文面试题·c++面经·c++面试题
码界奇点1 分钟前
基于DDD与CQRS的Java企业级应用框架设计与实现
java·开发语言·c++·毕业设计·源代码管理
Frank_refuel3 分钟前
C++STL之set和map的接口使用介绍
数据库·c++·算法
java修仙传3 分钟前
力扣hot100:跳跃游戏||
算法·leetcode·游戏
闻缺陷则喜何志丹4 分钟前
【模拟】P9670 [ICPC 2022 Jinan R] Frozen Scoreboard|普及+
c++·算法·模拟·洛谷
永远都不秃头的程序员(互关)7 分钟前
【K-Means深度探索(十一)】K-Means VS 其他聚类算法:如何选择最合适的工具?
算法·kmeans·聚类
洛生&14 分钟前
Nested Ranges Count
算法
老鼠只爱大米14 分钟前
LeetCode经典算法面试题 #142:环形链表 II(哈希表、快慢指针等多种方法详细解析)
算法·leetcode·链表·快慢指针·floyd算法·环形链表
王老师青少年编程15 分钟前
2024年6月GESP真题及题解(C++八级): 最远点对
c++·题解·真题·gesp·csp·八级·最远点对
数智工坊16 分钟前
【操作系统-线程介绍】
linux·算法·ubuntu