LuoguP3627 [APIO2009]抢掠计划

P3627 [APIO2009]抢掠计划

题目描述

Siruseri 城中的道路都是单向的。不同的道路由路口连接。按照法律的规定, 在每个路口都设立了一个 Siruseri 银行的 ATM 取款机。令人奇怪的是,Siruseri 的酒吧也都设在路口,虽然并不是每个路口都设有酒吧。
Banditji 计划实施 Siruseri 有史以来最惊天动地的 ATM 抢劫。他将从市中心 出发,沿着单向道路行驶,抢劫所有他途径的 ATM 机,最终他将在一个酒吧庆 祝他的胜利。
使用高超的黑客技术,他获知了每个 ATM 机中可以掠取的现金数额。他希 望你帮助他计算从市中心出发最后到达某个酒吧时最多能抢劫的现金总数。他可 以经过同一路口或道路任意多次。但只要他抢劫过某个 ATM 机后,该 ATM 机 里面就不会再有钱了。 例如,假设该城中有 6 个路口,道路的连接情况如下图所示:

市中心在路口 1,由一个入口符号→来标识,那些有酒吧的路口用双圈来表示。每个 ATM 机中可取的钱数标在了路口的上方。在这个例子中,Banditji 能抢 劫的现金总数为 47,实施的抢劫路线是:1-2-4-1-2-3-5。

输入输出格式

输入格式:

第一行包含两个整数 N、M。N 表示路口的个数,M 表示道路条数。接下来 M 行,每行两个整数,这两个整数都在 1 到 N 之间,第 i+1 行的两个整数表示第 i 条道路的起点和终点的路口编号。接下来 N 行,每行一个整数,按顺序表示每 个路口处的 ATM 机中的钱数。接下来一行包含两个整数 S、P,S 表示市中心的 编号,也就是出发的路口。P 表示酒吧数目。接下来的一行中有 P 个整数,表示 P 个有酒吧的路口的编号。

输出格式:

输出一个整数,表示 Banditji 从市中心开始到某个酒吧结束所能抢劫的最多 的现金总数。

输入输出样例

输入样例#1:

6 7
1 2
2 3
3 5
2 4
4 1
2 6
6 5
10
12
8
16
1
5
1 4
4 3 5 6

输出样例#1:

47

说明

50%的输入保证 N, M<=3000。所有的输入保证 N, M<=500000。每个 ATM 机中可取的钱数为一个非负整数且不超过 4000。
输入数据保证你可以从市中心 沿着 Siruseri 的单向的道路到达其中的至少一个酒吧。

题解

tarjan缩点后,DP即可。DP[i] = max(DP[V]) + val2[V],注意重点必须是酒吧,因此转移时要看V能不能直接或间接到酒吧

#include <iostream> 
#include <cstdio>
#include <cstring>
#include <map>
#include <cstdlib>
#define max(a, b) ((a) > (b) ? (a) : (b))
#define min(a, b) ((a) < (b) ? (a) : (b))
template <class T>
void swap(T &a, T &b)
{
	T tmp = a;a = b;b = tmp;
}
inline void read(int &x)
{
	x = 0;char ch = getchar(), c = ch;
	while(ch < '0' || ch > '9') c = ch, ch = getchar();
	while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
	if(c == '-') x = -x; 
}
const int INF = 0x3f3f3f3f;
const int MAXN = 500000 + 10;
const int MAXM = 500000 + 10;
struct Edge
{
	int u,v,nxt;
	Edge(int _u, int _v, int _nxt){u = _u, v = _v, nxt = _nxt;}
	Edge(){}
}edge1[MAXM << 1], edge2[MAXM << 1];
int head1[MAXN], cnt1, head2[MAXN], cnt2;
inline void insert1(int a, int b){edge1[++cnt1] = Edge(a,b,head1[a]), head1[a] = cnt1;}
inline void insert2(int a, int b){edge2[++cnt2] = Edge(a,b,head2[a]), head2[a] = cnt2;}
int b[MAXN], bb[MAXN], low[MAXN], dfn[MAXN], dfst, stack[MAXN], top, belong[MAXN], val2[MAXN], val1[MAXN], group, n, m;
void dfs(int x)
{
	b[x] = bb[x] = 1, stack[++top] = x, low[x] = dfn[x] = ++ dfst;
	for(int pos = head1[x];pos;pos = edge1[pos].nxt)
	{
		int v = edge1[pos].v;
		if(!b[v]) dfs(v), low[x] = min(low[x], low[v]);
		else if(bb[v] && low[x] > dfn[v]) low[x] = dfn[v];
	}
	if(low[x] == dfn[x])
	{
		int now = -1;++ group;
		while(now != x) now = stack[top --], belong[now] = group, val2[group] += val1[now], bb[now] = 0;
	}
}
void rebuild()
{
	for(int i = 1;i <= cnt1;++ i)
		if(belong[edge1[i].u] != belong[edge1[i].v]) insert2(belong[edge1[i].u], belong[edge1[i].v]);
}
void tarjan()
{
	for(int i = 1;i <= n;++ i) if(!b[i]) dfs(i);
	rebuild();
}
int S, P, isp[MAXN], dp[MAXN], vis[MAXN];
int DP(int x)
{
	if(vis[x]) return dp[x];
	int tmp = 0;
	for(int pos = head2[x];pos;pos = edge2[pos].nxt)
	{
		int v = edge2[pos].v;
		tmp = max(tmp, DP(v));
		if(vis[v]) vis[x] = 1;
	}
	if(vis[x] == 0 && isp[x] == 1) vis[x] = 1;
	return vis[x] ? dp[x] = tmp + val2[x] : 0;
}
int main()
{
	read(n), read(m);
	for(int i = 1;i <= m;++ i)
	{
		int tmp1,tmp2;
		read(tmp1), read(tmp2);
		insert1(tmp1, tmp2);
	}
	for(int i = 1;i <= n;++ i) read(val1[i]);
	tarjan();
	read(S), read(P);
	for(int i = 1;i <= P;++ i)
	{
		int tmp;read(tmp);
		isp[belong[tmp]] = 1;
	}
	DP(belong[S]);
	printf("%d", dp[belong[S]]);
	return 0;
}
posted @ 2018-02-09 15:21  嘒彼小星  阅读(226)  评论(0编辑  收藏  举报