「LibreOJ NOIP Round #1」旅游路线
Description
T 城是一个旅游城市,具有 nnn 个景点和 mmm 条道路,所有景点编号为 1,2,...,n1,2,...,n1,2,...,n。每条道路连接这 nnn 个景区中的某两个景区,道路是单向通行的。每条道路都有一个长度。
为了方便旅游,每个景点都有一个加油站。第 iii 个景点的加油站的费用为 pip_ipi,加油量为 cic_ici。若汽车在第 iii 个景点加油,则需要花费 pip_ipi 元钱,之后车的油量将被加至油量上限与 cic_ici 中的较小值。不过如果加油前汽车油量已经不小于 cic_ici,则不能在该景点加油。
小 C 准备来到 T 城旅游。他的汽车油量上限为 CCC。旅游开始时,汽车的油量为 000。在旅游过程中:
1、当汽车油量大于 000 时,汽车可以沿从当前景区出发的任意一条道路到达另一个景点(不能只走道路的一部分),汽车油量将减少 111;
2、当汽车在景点 iii 且当前油量小于 cic_ici 时,汽车可以在当前景点加油,加油需花费 pip_ipi 元钱,这样汽车油量将变为 min{ci,C}\min{c_i,C}min{ci,C}。
一次旅游的总花费等于每次加油的花费之和,旅游的总路程等于每次经过道路的长度之和。注意多次在同一景点加油,费用也要计算多次,同样地,多次经过同一条道路,路程也要计算多次。
小 C 计划旅游 TTT 次,每次旅游前,小 C 都指定了该次旅游的起点和目标路程。由于行程不同,每次出发前带的钱也不同。为了省钱,小 C 需要在旅游前先规划好旅游路线(包括旅游的路径和加油的方案),使得从起点出发,按照该旅游路线旅游结束后总路程不小于目标路程,且剩下的钱尽可能多。请你规划最优旅游路线,计算这 TTT 次旅游每次结束后最多可以剩下多少钱。
solution
看到 \(dis>=10^9\) 和 \(n<=100\) 这种东西,要想到倍增Floyed,首先要发现我们不能把油记录在dp状态里面否则开不下,我们考虑枚举加油位置.
我们定义 \(dp[i][j]\) 表示从 \(i\)出发,在j加满油,花费 \(j\) 元钱可以走的最大距离,答案就是满足 \(dp[i][j]>=d\) 的最大j,所以我们要想办法预处理出这个,然后询问就可以快速回答了,\(dp[i][j]=Max(dp[k][j-p[i]]+dis[j][k])\),\(dis[i][j]\) 表示从 \(i\) 出发达到 \(j\),经过道路数不超过 \(Min(c[i],C)\) 的最长距离,可以倍增Floyd预处理出来,然后dp数组具有二分性,可以二分回答询问,但是我爆枚也过了,最坏10^9,常数太小了,哎...
#include<iostream>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<vector>
#include<queue>
#include<cstring>
#define Max(a,b) ((a)>(b)?(a):(b))
#define Min(a,b) ((a)<(b)?(a):(b))
#define RG register
using namespace std;
typedef long long ll;
const int N=105,M=2005;
const ll inf=2e15;
int n,m,C,T,p[N],c[N];ll w[N][N][20];
ll f[N],g[N],dis[N][N],dp[N][N*N];
void work()
{
scanf("%d%d%d%d",&n,&m,&C,&T);
for(int i=1;i<=n;i++){
scanf("%d%d",&p[i],&c[i]);
if(c[i]>C)c[i]=C;
}
int x,y,z;
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
for(int k=0;k<=18;k++)
w[i][j][k]=-inf;
for(int i=1;i<=n;i++)w[i][i][0]=0;
for(int i=1;i<=m;i++){
scanf("%d%d%d",&x,&y,&z);
w[x][y][0]=Max(w[x][y][0],z);
}
for(int k=1;k<=18;k++)
for(int l=1;l<=n;l++)
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
w[i][j][k]=Max(w[i][j][k],w[i][l][k-1]+w[l][j][k-1]);
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++)
g[j]=(i==j?0:-inf),f[j]=-inf;
for(int k=18;k>=0;k--){
if(((1<<k)&c[i])==0)continue;
for(int j=1;j<=n;j++){
for(int l=1;l<=n;l++)
f[j]=Max(f[j],g[l]+w[l][j][k]);
}
for(int j=1;j<=n;j++)g[j]=f[j];
}
for(int j=1;j<=n;j++)dis[i][j]=g[j];
}
int lim=n*n;
for(int k=0;k<=lim;k++){
for(int i=1;i<=n;i++){
if(k<p[i])continue;
for(int j=1;j<=n;j++)
dp[i][k]=Max(dp[i][k],dp[j][k-p[i]]+dis[i][j]);
}
}
while(T--){
scanf("%d%d%d",&x,&y,&z);
int ans=y+1;
for(int i=0;i<=y;i++){
if(dp[x][i]>=z){ans=i;break;}
}
printf("%d\n",y-ans);
}
}
int main()
{
freopen("trip.in","r",stdin);
freopen("trip.out","w",stdout);
work();
return 0;
}
「LibreOJ NOIP Round #1」旅游路线的更多相关文章
- LibreOJ #539. 「LibreOJ NOIP Round #1」旅游路线(倍增+二分)
哎一开始看错题了啊T T...最近状态一直不对...最近很多傻逼题都不会写了T T 考虑距离较大肯定不能塞进状态...钱数<=n^2能够承受, 油量再塞就不行了...显然可以预处理出点i到j走c ...
- 【LibreOJ】#539. 「LibreOJ NOIP Round #1」旅游路线
[题意]给定正边权有向图,车油量上限C,每个点可以花费pi加油至min(C,ci),走一条边油-1,T次询问s点出发带钱q,旅行路程至少为d的最多剩余钱数. n<=100,m<=1000, ...
- LOJ#539. 「LibreOJ NOIP Round #1」旅游路线
n<=100,m<=1000的图,在此图上用油箱容量C<=1e5的车来旅行,旅行时,走一条边会耗一单伟油,在点i时,若油量<ci,则可以把油以pi的价格补到ci,pi<= ...
- LOJ #539. 「LibreOJ NOIP Round #1」旅游路线 倍增floyd + 思维
考试的时候是这么想的: 求出每一个点花掉 $i$ 的花费向其他点尽可能走的最长距离,然后二分这个花费,找到第一个大于 $d$ 的就输出$.$然而,我这个记忆化搜索 $TLE$ 的很惨$.$这里讲一下正 ...
- 「LOJ 539」「LibreOJ NOIP Round #1」旅游路线
description 题面较长,这里给出题目链接 solution 考虑预处理出\(f[i][j]\)表示在第\(i\)个点加满油后,从第\(i\)个点出发,至多消耗\(j\)元钱走过的最大路程,那 ...
- LibreOj #539. 「LibreOJ NOIP Round #1」旅游路线
题目链接 做完这道题,我深知当一个问题复杂度过高的时候,把一些可以分离的操作都分散开,可以大幅度降低复杂度..... 发现无论有多少钱,每到一个点后扩展到的距离被限制在 \(min(C, c[i])\ ...
- 【LibreOJ】#538. 「LibreOJ NOIP Round #1」数列递推
[题意]LibreOJ [算法]乱搞 [题解]容易发现数列最后一定单调,最后单调递增则最大值赋为最后一个,反之最小值赋为最后一个,然后处理一些细节就可以AC,要注意以下几点: 1.数列连续三项以及数列 ...
- 题解【loj537】「LibreOJ NOIP Round #1」DNA 序列
题目描述 \(NOIP\)复赛之前\(HSD\)桑进行了一项研究,发现人某条染色体上的一段\(DNA\)序列中连续的\(k\)个碱基组成的碱基序列与做题的 \(AC\) 率有关!于是他想研究一下这种关 ...
- 「LOJ 537」「LibreOJ NOIP Round #1」DNA 序列
description NOIP 复赛之前,HSD 桑进行了一项研究,发现人某条染色体上的一段 DNA 序列中连续的\(k\)个碱基组成的碱基序列与做题的 AC 率有关!于是他想研究一下这种关系. 现 ...
随机推荐
- 设计模式NO.3
设计模式NO.3 本次博客内容为第三次设计模式的练习.根据老师的要求完成下列题目: 题目1 某商品管理系统的商品名称存储在一个字符串数组中,现需要自定义一个双向迭代器(MyIterator)实现对该商 ...
- Beta冲刺Day2
项目进展 李明皇 今天解决的进度 优化了信息详情页的布局:日期显示,添加举报按钮等 优化了程序的数据传递逻辑 明天安排 程序运行逻辑的完善 林翔 今天解决的进度 实现微信端消息发布的插入数据库 明天安 ...
- nyoj 苹果
苹果 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 ctest有n个苹果,要将它放入容量为v的背包.给出第i个苹果的大小和价钱,求出能放入背包的苹果的总价钱最大值. ...
- C#中DBNull.Value和Null的用法和区别
DBNull.Value,, 是适用于向数据库的表中插入空值.而 null,是指在程序中表示空引用. 或者对象为空.就是没有实例化. row[column]的值为DBNull.Value的话,至少说明 ...
- js回顾(DOM中标签的CRUD,表格等)
01-DOM中的创建和添加标签 02-删除替换克隆标签 03-全选全不选反选 04-新闻字体 05-表格增删 06-动态生成表格 07-表格隔行变色 08-左到右右到左(将左边的标签移动到右边) 09 ...
- (java基础)Java输入输出流及文件相关
字节流: 所有的字节输入输出都继承自InputStream和OutputStream,通常用于读取二进制数据,最基本单位为单个字节,如图像和声音.默认不使用缓冲区. FileInputStream和F ...
- IIFE的形式、原理和常见写法
immediately invoked function expression 立即调用的函数表达式 出现的形式: 最近做牛客网的JS在线编程,遇到几道考闭包的题目,发现闭包的尾巴那儿经常会出现一对小 ...
- 论文泛读·Adversarial Learning for Neural Dialogue Generation
原文翻译 导读 这篇文章的主要工作在于应用了对抗训练(adversarial training)的思路来解决开放式对话生成(open-domain dialogue generation)这样一个无监 ...
- 初识JavaScript闭包
一个问题引发的思考 在我学习javascript的事件时,有一个小任务是使用JS来实现 li 列表项在鼠标悬浮时会有背景阴影的动态效果,很自然想到用for 来为每个列表项添加onmouseover 和 ...
- A、B同时打开一个页面进行同一条数据库记录进行修改,A修改完成后提交表单,A修改的数据保存完成后;当B也修改完成后,提交数据进行数据修改。此时B修改的内容会覆盖A修改的内容,请问如何避免?
A.B同时打开一个页面进行数据中的一条数据进行修改,A修改完成后提交表单,数据修改保存完成后B开始页面也修改完成,开始提交进行修改.此时B修改的内容会覆盖A的内容,请问如何避免? 通过搜索和我个人总结 ...