https://codeforc.es/contest/1137/problem/C

# 题意

给你n个点,每个点有k天博物馆开放时间的安排表。

有m条单向道路,走过一条边需要一个晚上,经过后就是第二天的意思。

问在无穷大的时间里,可以参观多少不同的博物馆。

# 思路

我们把每个点都拆出k个点,有单向边相连就从(u,i) -> (v, (i+1)%k)。

缩点跑出DAG,然后DP出最多的博物馆参观数。

这里就要考虑直接DP是否能行。假设有一条(u,i) -> (u, j)的边,由于没有自环且是单向边,所以一定可以通过循环,可以从(u,j)再走到(u,i),所以这两个点会缩在一起。

由于这种做法缩点时递归很深,我是通过inline优化的,开O(3)好像也可以。

#include <bits/stdc++.h>
using namespace std;
#define pb push_back
#define fi first
#define se second
#define debug(x) cerr<<#x << " := " << x << endl;
#define bug cerr<<"-----------------------"<<endl;
#define FOR(a, b, c) for(int a = b; a <= c; ++ a)
 
typedef long long ll;
typedef long double ld;
typedef pair<int, int> pii;
typedef pair<ll, ll> pll;
 
 
template<class T> void _R(T &x) { cin >> x; }
void _R(int &x) { scanf("%d", &x); }
void _R(ll &x) { scanf("%lld", &x); }
void _R(double &x) { scanf("%lf", &x); }
void _R(char &x) { scanf(" %c", &x); }
void _R(char *x) { scanf("%s", x); }
void R() {}
template<class T, class... U> void R(T &head, U &... tail) { _R(head); R(tail...); }
 
 
template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
}
 
const int inf = 0x3f3f3f3f;
 
const int mod = 1e9+;
 
/**********showtime************/
const int maxn = ;
 
// vector<int>mp[maxn];
 
int n,m,k;
int getid(int x, int i) {
return (x - ) * k + i + ;
}
 
struct E{
int u,v;
int nxt;
}edge[maxn];
 
int gtot = , head[maxn];
 
void addedge(int u, int v) {
edge[gtot].u = u;
edge[gtot].v = v;
edge[gtot].nxt = head[u];
head[u] = gtot++;
}
 
set <int> dpmp[maxn];
 
char str[];
int dp[maxn],a[maxn];
int belong[maxn], dfn[maxn] , low[maxn];
 
bool vis[maxn];
bool flag[maxn];
// int used[maxn];
 
int tim, scc_cnt;
 
//stack<int>st;
queue<int>que;
int st[];
int top = ;
inline void dfs(int u) {
dfn[u] = low[u] = ++tim;
 
// st.push(u);
st[++top] = u;
for(int i=head[u]; ~i; i=edge[i].nxt){
int v = edge[i].v;
if(!dfn[v]) dfs(v);
if(!belong[v]) low[u] = min(low[u], low[v]);
}
 
if(dfn[u] == low[u]) {
scc_cnt++;
int now;
while(true) {
now = st[top]; top--;
belong[now] = scc_cnt;
if(flag[now]) {
int id = (now - ) / k + ;
if(vis[id] == ) {
que.push(id);
vis[id] = ;
a[scc_cnt]++;
}
}
if(now == u) break;
}
while(!que.empty()) {
int u = que.front(); que.pop();
vis[u] = ;
}
}
}
 
int ans = ; void cal(int s) {
queue<int>que;
dp[s] = a[s];
que.push(s);
ans = max(ans, a[s]);
 
while(!que.empty()) {
int u = que.front(); que.pop();
vis[u] = ;
for(int v : dpmp[u]) {
dp[v] = max(dp[v], dp[u] + a[v]);
ans = max(ans, dp[v]);
if(vis[v] == ) {
vis[v] = ;
que.push(v);
}
}
}
}
int main(){
memset(head, -, sizeof(head));
R(n, m, k);
for(int i=; i<=m; i++) {
int u,v;
scanf("%d%d", &u, &v);
for(int j=; j<k; j++) {
addedge(getid(u, j), getid(v, (j+)%k));
}
}
 
for(int i=; i<=n; i++) {
scanf("%s", str);
for(int j=; j<k; j++) {
flag[getid(i, j)] = (str[j] == '');
}
}
 
for(int i=; i<=n; i++){
for(int j=; j<k; j++)
if(!dfn[getid(i, j)]) dfs(getid(i, j));
}
 
for(int i=; i<gtot; i++){
int u = edge[i].u, v = edge[i].v;
if(belong[u] == belong[v]) continue;
dpmp[belong[u]].insert(belong[v]);
}
 
cal(belong[getid(, )]);
 
printf("%d\n", ans);
return ;
}

【CF1137C】 Museums Tour 拆点+缩点的更多相关文章

  1. CF1137C Museums Tour(Tarjan,强连通分量)

    好题,神题. 题目链接:CF原网 洛谷 题目大意: 一个国家有 $n$ 个城市,$m$ 条有向道路组成.在这个国家一个星期有 $d$ 天,每个城市有一个博物馆. 有个旅行团在城市 $1$ 出发,当天是 ...

  2. CF1137C Museums Tour(tarjan+DP)

    由于d很小,所以可以把每个点拆成d个点,然后对于边(x,y),连边时连接((x,i),(y,i+1))及((x,d),(y,1)).然后可以对这样连的边跑一遍tarjan缩点.然后直接暴力DP即可.不 ...

  3. CF1137C Museums Tour

    思路 强连通分量的好题 对于每个博物馆,因为时间的限制条件,不好直接统计, 发现d很小,可以建出d层分层图,原图<u,v>的边变成<u,i>到<v,i+1>的边,& ...

  4. CF1137 C. Museums Tour

    CF1137 C. Museums Tour 一般来说的正常思路:看到有向图的第一思路都是缩点(但是要分析一波证明强联通分量中的个体可以拼凑成整体,一般都是边和点可以经过无数次然后贡献只算一次这种类型 ...

  5. CF 1138 E. Museums Tour

    E. Museums Tour 链接 分析: 按时间建出分层图,每个点形如(u,t),表示u在在t个时刻的点,tarjan缩点.每个强连通分量中的点都能经过,然后DAG上dp. 代码: #includ ...

  6. hdu3488 Tour 拆点+二分图最佳匹配

    In the kingdom of Henryy, there are N (2 <= N <= 200) cities, with M (M <= 30000) one-way r ...

  7. 【Codeforces 1137C】Museums Tour

    Codeforces 1137 C 题意:给一个有向图,一周有\(d\)天,每一个点在每一周的某些时刻会开放,现在可以在这个图上从\(1\)号点开始随意地走,问最多能走到多少个开放的点.一个点如果重复 ...

  8. [CF1137]Museums Tour

    link \(\text{Description:}\) 一个国家有 \(n\) 个城市,\(m\) 条有向道路组成.在这个国家一个星期有 \(d\) 天,每个城市有一个博物馆. 有个旅行团在城市 \ ...

  9. Codeforces 1137C Museums Tour (强连通分量, DP)

    题意和思路看这篇博客就行了:https://www.cnblogs.com/cjyyb/p/10507937.html 有个问题需要注意:对于每个scc,只需要考虑进入这个scc的时间即可,其实和从哪 ...

随机推荐

  1. python Django编写接口并用Jmeter测试

    一.环境准备 python3.6.7 Pycharm 二.创建项目 我这里是在Django项目中新建了个APP,目录结构如下图所示: 那么怎么在已有的Django项目中新建APP并进行配置呢: 2.1 ...

  2. 两份简单的logstash配置

    input{http{port=>7474}} filter{ grok{ match =>{ #"message" => "%{COMBINEDAPA ...

  3. Drawable与 Bitmap 转换总结

    极力推荐文章:欢迎收藏 Android 干货分享 阅读五分钟,每日十点,和您一起终身学习,这里是程序员Android Drawable 使用方法详解请看上篇文章. Drawable 使用方法详解 本篇 ...

  4. Powered by .NET Core 进展:用 docker-compose 验证高并发问题嫌疑犯 docker swarm

    相关博文: [故障公告]发布 .NET Core 版博客站点引起大量 500 错误 [网站公告].NET Core 版博客站点第二次发布尝试 暴风雨中的 online : .NET Core 版博客站 ...

  5. nginx在线与离线安装

    1.场景描述 项目要部署到新的服务器上,需要安装nginx,刚好安全部门通知了nginx存在安全漏洞(Nginx整数溢出漏洞,nginx1.13.2之后的版本无问题),就下载最新的nginx进行了安装 ...

  6. Java 操作Word书签(一):添加、删除、读取书签

    Word中,书签功能常用于查找.定位.标记特定字符或段落,对于篇幅较大的文档,此功能非常实用.下面,将介绍通过Java程序来添加及删除Word书签的方法.示例要点包括: 1. 添加书签 1.1 给指定 ...

  7. Nunit与Xunit介绍

    Nunit安装 首先说下,nunit2.X与3.X版本需要安装不同的vs扩展. nunit2.x安装 安装如上3个,辅助创建nunit测试项目与在vs中运行单元测试用例 . 1.Nunit2 Test ...

  8. centos7单机安装kafka,进行生产者消费者测试

    [转载请注明]: 原文出处:https://www.cnblogs.com/jstarseven/p/11364852.html   作者:jstarseven    码字挺辛苦的.....  一.k ...

  9. Android8.1 MTK平台 截屏功能分析

    前言 涉及到的源码有 frameworks\base\services\core\java\com\android\server\policy\PhoneWindowManager.java vend ...

  10. Elasticsearch6.x和7.x版本常用插件汇总

    elasticsearch插件汇总 基于es 7.3版本试用. 一.安全插件 1.x-pack a.介绍 包括安全(x-pack-security),监视(x-pack-watcher),警报(x-p ...