BZOJ1924 [Sdoi2010]所驼门王的宝藏 【建图 + tarjan】
题目

输入格式
第一行给出三个正整数 N, R, C。 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室,类型为 Ti。Ti是一个1~3间的整数, 1表示可以传送到第 xi行任意一列的“横天门”,2表示可以传送到任意一行第 yi列的“纵寰门”,3表示可以传送到周围 8格宫室的“ziyou门”。 保证 1≤xi≤R,1≤yi≤C,所有的传送门位置互不相同。
输出格式
只有一个正整数,表示你确定的路线所经过不同藏宝宫室的最大数目。
输入样例
10 7 7
2 2 1
2 4 2
1 7 2
2 7 3
4 2 2
4 4 1
6 7 3
7 7 1
7 5 2
5 2 1
输出样例
9
提示

题解
把图建出来后就是tarjan + dp了
建图比较烦
我们将其按x排序,先将横向边建上,当一排存在多个横向边时,它们之间两两连边,过于费空间,只连一个环就好了
竖向边也是一样的
最后是第三种边,排个序二分查找就好了
#include<iostream>
#include<cstdio>
#include<cmath>
#include<queue>
#include<vector>
#include<cstring>
#include<algorithm>
#define LL long long int
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define BUG(s,n) for (int i = 1; i <= (n); i++) cout<<s[i]<<' '; puts("");
using namespace std;
const int maxn = 100005,maxm = 1000005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
int n,N,M;
int x[maxn],y[maxn],id[maxn],t[maxn];
inline bool cmp1(const int& a,const int& b){
return x[a] == x[b] ? t[a] < t[b] : x[a] < x[b];
}
inline bool cmp2(const int& a,const int& b){
return y[a] == y[b] ? t[a] < t[b] : y[a] < y[b];
}
inline bool cmp3(const int& a,const int& b){
return x[a] == x[b] ? y[a] < y[b] : x[a] < x[b];
}
int h[maxn],ne = 2,h2[maxn],ne2 = 2;
struct EDGE{int to,nxt;}ed[maxm],e[maxm];
inline void build(int u,int v){
ed[ne] = (EDGE){v,h[u]}; h[u] = ne++;
//printf("build (%d) to (%d)\n",u,v);
}
inline void add(int u,int v){
e[ne2] = (EDGE){v,h2[u]}; h2[u] = ne2++;
}
inline bool check(const int& p,const int& X,const int& Y){
return x[id[p]] == X ? y[id[p]] >= Y : x[id[p]] > X;
}
int lowb(int X,int Y){
int l = 1,r = n,mid;
while (l < r){
mid = l + r >> 1;
if (check(mid,X,Y)) r = mid;
else l = mid + 1;
}
return l;
}
int st[maxn],top;
void Build(){
sort(id + 1,id + 1 + n,cmp1);
for (int i = 1; i <= n; i++){
int u = i; top = 0;
while (u <= n){
if (x[id[u]] != x[id[i]]) break;
if (t[id[u]] == 1) st[++top] = id[u];
u++;
}
if (!top) continue;
if (top > 1){
for (int j = 1; j < top; j++) build(st[j],st[j + 1]);
build(st[top],st[1]);
}
for (int j = i; j < u; j++) if (t[id[j]] != 1) build(st[1],id[j]);
i = --u;
}
sort(id + 1,id + 1 + n,cmp2);
for (int i = 1; i <= n; i++){
int u = i; top = 0;
while (u <= n){
if (y[id[u]] != y[id[i]]) break;
if (t[id[u]] == 2) st[++top] = id[u];
u++;
}
if (!top) continue;
if (top > 1){
for (int j = 1; j < top; j++) build(st[j],st[j + 1]);
build(st[top],st[1]);
}
for (int j = i; j < u; j++) if (t[id[j]] != 2) build(st[1],id[j]);
i = --u;
}
sort(id + 1,id + 1 + n,cmp3);
for (int i = 1; i <= n; i++){
if (t[id[i]] != 3) continue;
int u = lowb(x[id[i]] - 1,y[id[i]] - 1);
for (int j = u; j <= n && x[id[j]] + 1 == x[id[i]] && y[id[j]] >= y[id[i]] - 1 && y[id[j]] <= y[id[i]] + 1; j++)
build(id[i],id[j]);
u = lowb(x[id[i]],y[id[i]] - 1);
for (int j = u; j <= n && x[id[j]] == x[id[i]] && y[id[j]] >= y[id[i]] - 1 && y[id[j]] <= y[id[i]] + 1; j++)
if (id[i] != id[j]) build(id[i],id[j]);
u = lowb(x[id[i]] + 1,y[id[i]] - 1);
for (int j = u; j <= n && x[id[j]] - 1 == x[id[i]] && y[id[j]] >= y[id[i]] - 1 && y[id[j]] <= y[id[i]] + 1; j++)
build(id[i],id[j]);
}
}
int dfn[maxn],low[maxn],Scc[maxn],scci,cnt,val[maxn];
void dfs(int u){
dfn[u] = low[u] = ++cnt;
st[++top] = u;
Redge(u){
if (!dfn[to = ed[k].to]){
dfs(to);
low[u] = min(low[u],low[to]);
}else if (!Scc[to]) low[u] = min(low[u],dfn[to]);
}
if (dfn[u] == low[u]){
scci++;
do{
Scc[st[top]] = scci; val[scci]++;
}while(st[top--] != u);
}
}
void tarjan(){
for (int i = 1; i <= n; i++) if (!dfn[i]) dfs(i);
}
queue<int> q;
int inde[maxn],f[maxn];
void solve(){
for (int i = 1; i <= n; i++){
int u = Scc[i];
Redge(i) if (Scc[to = ed[k].to] != u)
add(u,Scc[to]),inde[Scc[to]]++;
}
for (int i = 1; i <= scci; i++) if (!inde[i]) q.push(i),f[i] = val[i];
int u,ans = 0;
while (!q.empty()){
u = q.front(); q.pop();
ans = max(ans,f[u]);
for (int k = h2[u],to; k; k = e[k].nxt){
f[to = e[k].to] = max(f[to],f[u] + val[to]);
if (!(--inde[to])) q.push(to);
}
}
for (int i = 1; i <= scci; i++)
ans = max(ans,f[i]);
printf("%d\n",ans);
}
void readin(){
n = read(); N = read(); M = read();
for (int i = 1; i <= n; i++)
x[i] = read(),y[i] = read(),t[i] = read(),id[i] = i;
}
int main(){
readin();
Build();
tarjan();
solve();
return 0;
}
BZOJ1924 [Sdoi2010]所驼门王的宝藏 【建图 + tarjan】的更多相关文章
- [SDOI2010] 所驼门王的宝藏 [建图+tarjan缩点+DAG dp]
题面传送门: 传送门 思路: 看完题建模,容易得出是求单向图最长路径的问题 那么把这张图缩强联通分量,再在DAG上面DP即可 然而 这道题的建图实际上才是真正的考点 如果对于每一个点都直接连边到它所有 ...
- BZOJ1924:[SDOI2010]所驼门王的宝藏(强连通分量,拓扑排序)
Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...
- bzoj1924: [Sdoi2010]所驼门王的宝藏
陈年老题又来水一发啊啊啊 构图狗了一点,然后其实强连通缩点dij找最长路就没了. 没调出来有点气,直接打了第9个点的表.... 来逛blog的你教教我呗 #include<cstdio> ...
- 【BZOJ-1924】所驼门王的宝藏 Tarjan缩点(+拓扑排序) + 拓扑图DP
1924: [Sdoi2010]所驼门王的宝藏 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 787 Solved: 318[Submit][Stat ...
- [BZOJ 1924][Sdoi2010]所驼门王的宝藏
1924: [Sdoi2010]所驼门王的宝藏 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 1285 Solved: 574[Submit][Sta ...
- 【题解】SDOI2010所驼门王的宝藏(强连通分量+优化建图)
[题解]SDOI2010所驼门王的宝藏(强连通分量+优化建图) 最开始我想写线段树优化建图的说,数据结构学傻了233 虽然矩阵很大,但是没什么用,真正有用的是那些关键点 考虑关键点的类型: 横走型 竖 ...
- [SDOI2010]所驼门王的宝藏
题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为"先知"的Alpaca L. Sotomon是这个家族的领袖,外人也称其为"所驼门王". ...
- [LuoguP2403][SDOI2010]所驼门王的宝藏
题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为"先知"的Alpaca L. Sotomon是这个家族的领袖,外人也称其为"所驼门王". ...
- BZOJ 1924: [Sdoi2010]所驼门王的宝藏 【tarjan】
Description 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为“先 知”的Alpaca L. Sotomon 是这个家族的领袖,外人也称其为“所驼门王”.所 驼门王毕生致力于维 ...
随机推荐
- JavaScript数据格式验证探讨
1.需求 修改某个文本框数据,要求对修改后的格式做验证(必须是数字). 注:实际需求比上述复杂,为了说明问题,这里特意简化了需求(如:对修改后数据依赖条件的判断,数据入库等). 2.关于NaN的探讨( ...
- SQLServer 错误: 15404,无法获取有关 Windows NT 组/ 用户 'WIN-8IVSNAQS8T7\Administrator' 的信息,错误代码 0x534。
在自动清理日志的作业中,执行过程出现如下问题:“SQLServer 错误: 15404,无法获取有关 Windows NT 组/ 用户 'WIN-8IVSNAQS8T7\Administrator' ...
- 清理ThreadLocal
在我很多的课程里(master.concurrency.xj-conc-j8),我经常提起ThreadLocal.它经常受到我严厉的指责要尽可能的避免使用.ThreadLocal是为了那些使用完就销毁 ...
- halcon相机标定及图像矫正
https://blog.csdn.net/humanking7/article/details/44756073 相机标定内容详解:转载自 祥的博客 预备知识 标定中的四个坐标系 1.1.平面旋转 ...
- mac ssd开启trim模式
开启方法 sudo trimforce enable
- Codeforces Round #318 (Div. 2) C Bear and Poker (数学)
简单题,求一下所有数的2和3的幂是任意调整的,把2和3的因子除掉以后必须相等. 求lcm,爆了long long.我得好好反省一下,对连乘不敏感 #include<bits/stdc++.h&g ...
- CPP-基础:内存泄露及其检测工具
[转]浅谈C/C++内存泄露及其检测工具 对于一个c/c++程序员来说,内存泄漏是一个常见的也是令人头疼的问题.已经有许多技术被研究出来以应对这个问题,比如 Smart Pointer,Garba ...
- Ubuntu下Hyperledger Fabric v0.6安装部署
系统环境:虚拟机VMware Workstation中的Ubuntu 16.04LTS 1.环境准备 1.1安装Docker Docker安装命令: curl –fsSL https://get.do ...
- windows下pycharm使用Anaconda安装包环境
转自: https://www.cnblogs.com/heitaoq/p/8632315.html
- Python可变与不可变类型及垃圾回收机制
1. 可变与不可变类型 1.1 可变类型 在id不变的情况下,value可以改变,则称之为可变类型.列表.字典与集合是可变的. l1 = [,,,,] print(id(l1)) l1[] = #改 ...