### 洛谷 P2656 题目链接 ###

题目大意:

小胖和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)的更多相关文章

  1. 动态规划 洛谷P4017 最大食物链计数——图上动态规划 拓扑排序

    洛谷P4017 最大食物链计数 这是洛谷一题普及/提高-的题目,也是我第一次做的一题 图上动态规划/拓扑排序 ,我认为这题是很好的学习拓扑排序的题目. 在这题中,我学到了几个名词,入度,出度,及没有环 ...

  2. zoj1232Adventure of Super Mario(图上dp)

    题目连接: 啊哈哈.点我点我 思路: 这个题目是一个图上dp问题.先floyd预处理出图上全部点的最短路,可是在floyd的时候,把可以用神器的地方预处理出来,也就是转折点地方不能为城堡..预处理完成 ...

  3. 洛谷 P6295 - 有标号 DAG 计数(生成函数+容斥+NTT)

    洛谷题面传送门 看到图计数的题就条件反射地认为是不可做题并点开了题解--实际上这题以我现在的水平还是有可能能独立解决的( 首先连通这个条件有点棘手,我们尝试把它去掉.考虑这题的套路,我们设 \(f_n ...

  4. BZOJ5291/洛谷P4458/LOJ#2512 [Bjoi2018]链上二次求和 线段树

    原文链接http://www.cnblogs.com/zhouzhendong/p/9031130.html 题目传送门 - LOJ#2512 题目传送门 - 洛谷P4458 题目传送门 - BZOJ ...

  5. 【BZOJ2763/洛谷p4563】【分层图最短路】飞行路线

    2763: [JLOI2011]飞行路线 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 4630  Solved: 1797[Submit][Stat ...

  6. Tarjan缩点+DAG图dp

    题目背景 缩点+DP 题目描述 给定一个n个点m条边有向图,每个点有一个权值,求一条路径,使路径经过的点权值之和最大.你只需要求出这个权值和. 允许多次经过一条边或者一个点,但是,重复经过的点,权值只 ...

  7. 洛谷P4458 /loj#2512.[BJOI2018]链上二次求和(线段树)

    题面 传送门(loj) 传送门(洛谷) 题解 我果然是人傻常数大的典型啊-- 题解在这儿 //minamoto #include<bits/stdc++.h> #define R regi ...

  8. 【洛谷 P4934】 礼物 (位运算+DP)

    题目链接 位运算+\(DP\)=状压\(DP\)?(雾 \(a\&b>=min(a,b)\)在集合的意义上就是\(a\subseteq b\) 所以对每个数的子集向子集连一条边,然后答案 ...

  9. 洛谷 P2015 二叉苹果树(codevs5565) 树形dp入门

    dp这一方面的题我都不是很会,所以来练(xue)习(xi),大概把这题弄懂了. 树形dp就是在原本线性上dp改成了在 '树' 这个数据结构上dp. 一般来说,树形dp利用dfs在回溯时进行更新,使用儿 ...

随机推荐

  1. Shell(2)—数组

    Shell(2)-数组 常用的 Bash Shell 只支持一维数组,不支持多维数组. 一.概念 Shell 并且没有限制数组的大小,理论上可以存放无限量的数据.Shell 数组元素的下标也是从 0 ...

  2. python库的tkinter带你进入GUI世界(计算器简单功能)

    前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者: 一个处女座的程序猿 PS:如有需要Python学习资料的小伙伴可以加 ...

  3. FCC---Learn How Bezier Curves Work---定义坐标轴点的值,影响斜率,改变速度。具体调试换值既可以体会

    The last challenge introduced the animation-timing-function property and a few keywords that change ...

  4. ADB控制手机命令(adb命令)

    手机端配置tcp方式连接 su setprop service.adb.tcp.port 5555 stop adbd start adbd 首先使用管理员权限,然后打开监听5555端口 电脑端使用a ...

  5. [转]Eclipse插件开发之基础篇(4) OSGi框架

    原文地址:http://www.cnblogs.com/liuzhuo/archive/2010/08/18/eclipse_plugin_1_2_1.html 1. 什么是OSGi框架 OSGi(O ...

  6. Flask 异步化

    web网站包含前端和后端, 异步处理可以用在前端, 也可以用在后端.  前端 jquery 进行 ajax 请求时, 可设置 async 属性为 true, 并为 success 设置一个 callb ...

  7. 6.python3实用编程技巧进阶(一)

    1.1.如何在列表中根据条件筛选数据 # 1.1.如何在列表中根据条件筛选数据 data = [-1, 2, 3, -4, 5] #筛选出data列表中大于等于零的数据 #第一种方法,不推荐 res1 ...

  8. 渗透测试学习 二十二、getshell总结

    大纲   管理员权限拿shell 普通权限拿shell 常见cms拿shell 进后台主要是可以对网站前台的内容,样式等做操作,要改脚本的内容的权限只有在webshell的权限下才可以(某些情况除外) ...

  9. Shell命令-网络操作之基础之ping、route

    文件及内容处理 - ping.route 1. ping:测试主机之间网络的连通性 ping命令的功能说明 ping 命令用于检测主机.执行 ping 指令会使用 ICMP 传输协议,发出要求回应的信 ...

  10. JAVA学习方法之——费曼学习法

    理查德·费曼 费曼简介 理查德·菲利普斯·费曼(Richard Phillips Feynman),出生于1918年5月11日,是美籍犹太裔物理学家,曾在1965年获得诺贝尔物理学奖,也被认为是继爱因 ...