【题意】给定正边权有向图,车油量上限C,每个点可以花费pi加油至min(C,ci),走一条边油-1,T次询问s点出发带钱q,旅行路程至少为d的最多剩余钱数。

n<=100,m<=1000,C<=10^5,q<=n^2

【算法】动态规划

【题解】官方题解

虽然不是DAG,但是由于q很小的特点,将q加入状态就满足DP的无后效性了。

令f[i][q]表示当前在i点并在i点加油,加油前钱数为q的最大路程。(q<pi时,f[i][q]=0)

假设下一加油点为j,转移方程:f[i][q]=max{f[j][q-pi]+w(i,j,ci)},其中w(i,j,c)表示i点到j点,油量为c的最大路程。(ci=min(ci,C))

现在主要问题是预处理w(i,j,ci),发现每走一条边c只会减少1,符合倍增变化规则一致的特点,考虑图上倍增。

虽然不是DAG,但是c加入状态就满足DP的无后效性了。

g(i,j,k)表示i点到j点,油量为2^k的最大路程,显然g(i,j,k)=max{g(i,x,k-1)+g(x,j,k-1)},x是中转点。

对于w(i,j,ci),将ci拆分二进制,每次枚举x作为中转点后直接取倍增数组g计算答案。

【倍增的思想是很经典的,将需要的ci拆分二进制后将1的位用倍增数组堆起来。但是应用到图上每次就都需要遍历全图作为可能的中转点,最后找到最优答案。】

最后得到了f[i][q],对每个询问在f[s]上二分到第一个大于等于d的f[s][q],q就是答案。

复杂度O(n^4+n^3*log k+T*log n^2),瓶颈复杂度O(n^4)。(n^4过100……)

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cctype>
using namespace std;
int read(){
char c;int s=,t=;
while(!isdigit(c=getchar()))if(c=='-')t=-;
do{s=s*+c-'';}while(isdigit(c=getchar()));
return s*t;
}
const int maxn=,maxm=,maxk=,inf=0x3f3f3f3f;
int n,m,C,T,tot;
int g[maxn][maxn][maxk],p[maxn],c[maxn],A[maxn],B[maxn],w[maxn][maxn],f[maxn][maxn*maxn];
void cmax(int &a,int b){if(b>a)a=b;}
int main(){
n=read();m=read();C=read();T=read();
for(int i=;i<=n;i++)for(int j=;j<=n;j++)for(int k=;k<=;k++)g[i][j][k]=-inf;
for(int i=;i<=n;i++)p[i]=read(),c[i]=min(C,read()),g[i][i][]=;
for(int i=;i<=m;i++){
int u=read(),v=read(),w=read();
g[u][v][]=w;
}
for(int k=;k<=;k++)
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
for(int x=;x<=n;x++)cmax(g[i][j][k],g[i][x][k-]+g[x][j][k-]);
for(int i=;i<=n;i++){
for(int j=;j<=n;j++)A[j]=-inf;A[i]=;
for(int k=;k<=;k++)if((c[i]>>k)&){
for(int j=;j<=n;j++){
B[j]=-inf;
for(int x=;x<=n;x++)cmax(B[j],A[x]+g[x][j][k]);
}
for(int j=;j<=n;j++)A[j]=B[j];
}
for(int j=;j<=n;j++)w[i][j]=A[j];
}
for(int q=;q<=n*n;q++)
for(int x=;x<=n;x++)if(q>=p[x])
for(int y=;y<=n;y++)cmax(f[x][q],f[y][q-p[x]]+w[x][y]);
while(T--){
int s=read(),q=read(),d=read();
int pl=lower_bound(f[s],f[s]+n*n+,d)-f[s];
if(pl>q)printf("-1\n");else printf("%d\n",q-pl);
}
return ;
}

最新文章

  1. 吉特仓库管理系统- 斑马打印机 ZPL语言的腐朽和神奇
  2. memset函数详解
  3. 第九天 内容提供者 ContentResolver
  4. [转](四)unity4.6Ugui中文教程文档-------概要-UGUI Visual Components
  5. oracle删除数据恢复
  6. 认识与学习 BASH
  7. SVN技术交流提纲
  8. TableInputFormat分片及分片数据读取源码级分析
  9. pptp记录用户登陆日志
  10. 嵌入式 hi3518平台指定网卡测试是否通外网
  11. CodeForces_#354_Div.2_2016.5.25(A+B+C)
  12. Notice
  13. PRD产品需求文档概要
  14. AzCopy – 上传/下载 Windows Azure Blob 文件
  15. ELK 完整部署和使用 - 每天5分钟玩转 Docker 容器技术(90)
  16. 简单检测PHP运行效率脚本
  17. day4:vcp考试
  18. python学习笔记05-列表
  19. Android通知栏介绍与适配总结(上篇)
  20. 阿里巴巴Java开发规约IDEA插件安装及使用

热门文章

  1. week12 201621044079 流与文件
  2. xpath教程二 ---- 通过ID和Class检索
  3. 2019 front end jobs collection
  4. Kubernetes初探 :总体概述及使用示例
  5. BZOJ 1509 逃学的小孩(树的直径)
  6. kafka搭建笔记
  7. [luogu5048] [Ynoi2019模拟赛] Yuno loves sqrt technology III
  8. async的基本用法
  9. React受控组件和非受控组件
  10. linux下,手动切换jdk