洛谷 P2656 (缩点 + DAG图上DP)
题目大意:
小胖和ZYR要去ESQMS森林采蘑菇。
ESQMS森林间有N个小树丛,M条小径,每条小径都是单向的,连接两个小树丛,上面都有一定数量的蘑菇。小胖和ZYR经过某条小径一次,可以采走这条路上所有的蘑菇。由于ESQMS森林是一片神奇的沃土,所以一条路上的蘑菇被采过后,又会长出一些新的蘑菇,数量为原来蘑菇的数量乘上这条路的“恢复系数”,再下取整。
比如,一条路上有4个蘑菇,这条路的“恢复系数”为0.7,则第一~四次经过这条路径所能采到的蘑菇数量分别为4,2,1,0.
现在,小胖和ZYR从S号小树丛出发,求他们最多能采到多少蘑菇。
分析:
1、题目没有限定路走的次数,显然如果图中有强连通分量(即有向环),那么这个环中的所有边都可以无数次访问,直到所有边能采到的蘑菇数量为 0 ,即为该连通分量的最大价值,故先进行缩点。
2、缩点之后,全图变为DAG图,然后每个点有价值,其次每条边有价值(且这个价值不会再乘以 恢复系数 ,因为缩点后不可能再走第二次)。那么求一些缩点与边权的最大价值,故用 DP 解答。
3、本题还要注意的是,它会给你一个起点,所以在拓扑排序中,在这个起点上面的点可以直接忽略,当然我们也可以全部初始化为负无穷,然后设起点 DP 值为该点的价值(如果这个起点是缩点的话,否则价值是 0 的),这样求最大值的时候,所有状态只会从起点转移,就不用将起点上面的点排除了,更方便。
代码如下:
#include<iostream>
#include<algorithm>
#include<string.h>
#include<queue>
#define maxn 80008
using namespace std;
int n,m,cnt,tot,Index,sum,st;
int low[maxn],dfn[maxn],flag[maxn],head[maxn],q[maxn];
int pre[maxn],s[maxn],in[maxn],qhead[maxn];
int b[maxn],dp[maxn];
bool vis[maxn];
struct Edge
{
int to;
int val;
double t;
int next;
}edge[];
struct EDGE
{
int to;
int val;
int next;
}E[];
inline void add(int u,int v,int w,double t)
{
edge[++cnt].to=v;
edge[cnt].val=w;
edge[cnt].t=t;
edge[cnt].next=head[u];
head[u]=cnt;
return;
}
inline void qadd(int u,int v,int w)
{
E[++cnt].to=v;
E[cnt].val=w;
E[cnt].next=qhead[u];
qhead[u]=cnt;
return;
}
void tarjan(int u)
{
low[u]=dfn[u]=++Index;
q[++tot]=u;
vis[u]=true;
for(int i=head[u];i;i=edge[i].next){
int v=edge[i].to;
if(!dfn[v]){
tarjan(v);
low[u]=min(low[u],low[v]);
}
else if(vis[v]) low[u]=min(low[u],dfn[v]);
}
if(low[u]==dfn[u])
{
++sum;//缩点的序号
do{
pre[q[tot]]=sum;
vis[q[tot--]]=false;
}while(q[tot+]!=u);
}
return;
}
void topo()
{
queue<int> Q;
while(!Q.empty()) Q.pop();
for(int i=;i<=sum;i++){
if(!in[i]) Q.push(i);
}
while(!Q.empty())
{
int x=Q.front();
Q.pop();
b[++cnt]=x,dp[x]=-0x3f3f3f3f;//记住要初始化为负无穷,不然会 WA 一个点
for(int i=qhead[x];i;i=E[i].next){
int v=E[i].to;
in[v]--;
if(!in[v]) Q.push(v);
}
}
return;
}
int main()
{
//freopen("test.in","r",stdin);
//freopen("test.out","w",stdout);
scanf("%d%d",&n,&m);
int A,B,C;
double D;
for(int i=;i<=m;i++){
scanf("%d%d%d%lf",&A,&B,&C,&D);
add(A,B,C,D);
}
scanf("%d",&st);
for(int i=;i<=n;i++){if(!dfn[i]) tarjan(i);}
cnt=;
for(int i=;i<=n;i++){
for(int j=head[i];j;j=edge[j].next){
int v=edge[j].to;
if(pre[i]!=pre[v]){qadd(pre[i],pre[v],edge[j].val);in[pre[v]]++;}
else{
int ans=edge[j].val;
int res=edge[j].val*edge[j].t;
while(res){ //算这个缩点的总价值,即按题意把所有边乘到为 0
ans+=res;
res=res*edge[j].t;
}
s[pre[i]]+=ans;
}
}
}
cnt=;
topo();//保证有向图 DP 顺序
dp[pre[st]]=s[pre[st]];//标记从起点所代表的缩点上为初始态转移
for(int i=;i<=cnt;i++){
int u=b[i];
for(int j=qhead[u];j;j=E[j].next){
int v=E[j].to;
dp[v]=max(dp[v],dp[u]+E[j].val+s[v]);
}
}
int ans=-0x3f3f3f3f;
for(int i=;i<=cnt;i++) ans=max(ans,dp[i]);
printf("%d\n",ans );
}
洛谷 P2656 (缩点 + DAG图上DP)的更多相关文章
- 动态规划 洛谷P4017 最大食物链计数——图上动态规划 拓扑排序
洛谷P4017 最大食物链计数 这是洛谷一题普及/提高-的题目,也是我第一次做的一题 图上动态规划/拓扑排序 ,我认为这题是很好的学习拓扑排序的题目. 在这题中,我学到了几个名词,入度,出度,及没有环 ...
- zoj1232Adventure of Super Mario(图上dp)
题目连接: 啊哈哈.点我点我 思路: 这个题目是一个图上dp问题.先floyd预处理出图上全部点的最短路,可是在floyd的时候,把可以用神器的地方预处理出来,也就是转折点地方不能为城堡..预处理完成 ...
- 洛谷 P6295 - 有标号 DAG 计数(生成函数+容斥+NTT)
洛谷题面传送门 看到图计数的题就条件反射地认为是不可做题并点开了题解--实际上这题以我现在的水平还是有可能能独立解决的( 首先连通这个条件有点棘手,我们尝试把它去掉.考虑这题的套路,我们设 \(f_n ...
- BZOJ5291/洛谷P4458/LOJ#2512 [Bjoi2018]链上二次求和 线段树
原文链接http://www.cnblogs.com/zhouzhendong/p/9031130.html 题目传送门 - LOJ#2512 题目传送门 - 洛谷P4458 题目传送门 - BZOJ ...
- 【BZOJ2763/洛谷p4563】【分层图最短路】飞行路线
2763: [JLOI2011]飞行路线 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 4630 Solved: 1797[Submit][Stat ...
- Tarjan缩点+DAG图dp
题目背景 缩点+DP 题目描述 给定一个n个点m条边有向图,每个点有一个权值,求一条路径,使路径经过的点权值之和最大.你只需要求出这个权值和. 允许多次经过一条边或者一个点,但是,重复经过的点,权值只 ...
- 洛谷P4458 /loj#2512.[BJOI2018]链上二次求和(线段树)
题面 传送门(loj) 传送门(洛谷) 题解 我果然是人傻常数大的典型啊-- 题解在这儿 //minamoto #include<bits/stdc++.h> #define R regi ...
- 【洛谷 P4934】 礼物 (位运算+DP)
题目链接 位运算+\(DP\)=状压\(DP\)?(雾 \(a\&b>=min(a,b)\)在集合的意义上就是\(a\subseteq b\) 所以对每个数的子集向子集连一条边,然后答案 ...
- 洛谷 P2015 二叉苹果树(codevs5565) 树形dp入门
dp这一方面的题我都不是很会,所以来练(xue)习(xi),大概把这题弄懂了. 树形dp就是在原本线性上dp改成了在 '树' 这个数据结构上dp. 一般来说,树形dp利用dfs在回溯时进行更新,使用儿 ...
随机推荐
- tf.contrib.slim模块简介
原文连接:https://blog.csdn.net/MOU_IT/article/details/82717745 1.简介 对于tensorflow.contrib这个库,tensorflow官方 ...
- three.js实现世界3d地图
概况如下: 1.THREE.Shape绘制世界地图平面地图: 2.THREE.ExtrudeGeometry将绘制的平面沿着Z轴拉伸,实现3d效果: 效果图如下: 预览地址:three.js实现世界3 ...
- Java内功心法,Set集合的详解
本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...
- Java生鲜电商平台-秒杀系统微服务架构设计与源码解析实战
Java生鲜电商平台-秒杀系统微服务架构设计与源码解析实战 Java生鲜电商平台- 什么是秒杀 通俗一点讲就是网络商家为促销等目的组织的网上限时抢购活动 比如说京东秒杀,就是一种定时定量秒杀,在规定 ...
- jpa分页
法一(本地sql查询,注意表名啥的都用数据库中的名称,适用于特定数据库的查询) public interface UserRepository extends JpaRepository<Use ...
- Javase之集合体系(3)之Set及其子类知识
集合体系之Set及其子类知识 Set(接口) public interface Set<E>extends Collection<E> 特点:无序(存储顺序与取出顺序不一致 ...
- JS基础语法---循环语句之:for 循环 + 9个练习
for循环 语法: for(表达式1;表达式2;表达式3){ 循环体; } 执行过程: 先执行一次表达式1,然后判断表达式2;如果不成立则直接跳出循环 如果表达式2成立,执行循环体的代码,结束后,跳到 ...
- PeriscopeHeartAnimation
// // ViewController.m // PeriscopeHeartAnimation // // Created by ldj on 4/28/15. // Copyright (c) ...
- 一文解读AI芯片之间的战争 (转)
2015年的秋天,北京的雨水比往年要多些,温度却不算太冷.这一年里,年仅23岁的姚颂刚刚拿到清华大学的毕业证书;32岁的陈天石博士毕业后已在中科院计算所待了整整8年;而在芯片界摸爬滚打了14年的老将何 ...
- Linux 解决Deepin深度系统无法在root用户启动Google Chrome浏览器的问题
解决Deepin无法在root用户启动Google Chrome浏览器的问题,步骤如下. 前提:如何用root用户登录系统?编辑 vim /etc/lightdm/lightdm.conf , 找到并 ...