【题解】SDOI2010所驼门王的宝藏(强连通分量+优化建图)
【题解】SDOI2010所驼门王的宝藏(强连通分量+优化建图)
最开始我想写线段树优化建图的说,数据结构学傻了233
虽然矩阵很大,但是没什么用,真正有用的是那些关键点
考虑关键点的类型:
- 横走型
- 竖走型
- 八连通型
本质上只有两种类型(走一大串/走八连通),我们考虑这样一种建图方法:
- 对于每一行每一列建立一个点(点权为\(0\))
- 对于关键点建立一个点(点权为\(1\))
然后考虑这样一种建图方式,得到一个有点权无边权图
- 关键点所在的行与列无偿地向这个关键点连边
- 横走型的关键点向行连一条边,竖走型同理
- 八连通型直接向周围的关键点连边
题目要求走到的点最多,也就是求一条最长路径,但是显然这个图上可能有正环,但是点权贡献只能算一次,自然想到直接缩点。缩完点后得到一个\(DAG\),直接在这个\(DAG\)上\(dp\)
\(dp_i\)表示从这个节点出发最长的路径,直接转移。
分析点数,显然是\(O(3n)\),分析边数,一个点最多连接十条边,\(tarjin\)是\(O(n)\)的,但是我们用了\(map\)所以复杂度\(O(n\log n)\),实际上,直接用unordered_map就是\(O(n)\)了,就帅一点...
至于实现,直接用\(map\)存std::pair < int ,int >就好了
//@winlere
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
using namespace std; typedef long long ll;
inline int qr(){
register int ret=0,f=0;
register char c=getchar();
while(c<48||c>57)f|=c==45,c=getchar();
while(c>=48&&c<=57) ret=ret*10+c-48,c=getchar();
return f?-ret:ret;
}
const int maxn=1e5+5;
struct E{
int to,nx;
E(){to=nx=0;}
E(const int&a,const int&b){to=a;nx=b;}
};
vector < E > e,e2;
pair < int ,int > node[maxn];
int head[maxn*3];
int head2[maxn*3];
int TT[maxn];
int dp[maxn*3];
int n,m,k;
void add2(const int&fr,const int&to){
e2.push_back(E(to,head2[fr]));
head2[fr]=e2.size()-1;
}
void add(const int&fr,const int&to){
e.push_back(E(to,head[fr]));
head[fr]=e.size()-1;
}
int idx[maxn],idy[maxn];
int w[maxn*3];
int stk[maxn*3];
int qaq,top;
int be[maxn*3];
int siz[maxn*3];
int dfn[maxn*3],low[maxn*3],in[maxn*3];
int tim,ans;
int qaqcnt;
int dfs2(const int&now){
if(dp[now]) return dp[now];
register int ret=0;
for(register int t=head2[now];t;t=e2[t].nx)
ret=max(ret,dfs2(e2[t].to));
return dp[now]=ret+siz[now];
}
void dfs(const int&now){
dfn[now]=low[now]=++tim;in[now]=1;stk[++top]=now;
for(register int t=head[now];t;t=e[t].nx){
if(!dfn[e[t].to])
dfs(e[t].to),low[now]=min(low[now],low[e[t].to]);
if(dfn[e[t].to]&&in[e[t].to])
low[now]=min(low[now],dfn[e[t].to]);
}
if(dfn[now]==low[now]){
register int temp=0;
++qaq;
do{
temp=stk[top--];
in[temp]=0;siz[qaq]+=w[temp];
be[temp]=qaq;
}while(temp!=now);
}
}
map < pair < int ,int > , int > s;
inline int init(const int&a,const int&b,const int&c){
e.push_back(E()); e.push_back(E());
e2.push_back(E()); e2.push_back(E());
qaqcnt=k=a;n=b;m=c;
for(register int t=1;t<=k;++t){
register int t1=qr(),t2=qr(),t3=qr();
node[t].first=t1;
node[t].second=t2;
TT[t]=t3; w[t]=1;
s[make_pair(t1,t2)]=t;
if(!idx[t1]) idx[t1]=++qaqcnt;
if(!idy[t2]) idy[t2]=++qaqcnt;
add(idx[t1],t);
add(idy[t2],t);
}
for(register int t=1;t<=k;++t){
if(TT[t]==1) add(t,idx[node[t].first]);
if(TT[t]==2) add(t,idy[node[t].second]);
if(TT[t]==3)
for(register int dx=-1;dx<=1;++dx)
for(register int dy=-1;dy<=1;++dy)
if(dx||dy){
auto f=s.find(make_pair(node[t].first+dx,node[t].second+dy));
if(f!=s.end()) add(t,f->second);
}
}
for(register int t=1;t<=qaqcnt;++t)
if(!dfn[t]) dfs(t);
for(register int t=1;t<=qaqcnt;++t)
for(register int i=head[t];i;i=e[i].nx)
if(be[e[i].to]!=be[t])
add2(be[t],be[e[i].to]);
for(register int t=1;t<=qaq;++t)
ans=max(ans,dfs2(t));
printf("%d\n",ans);
return 0;
}
int main(){
int a=qr(),b=qr(),c=qr();
return init(a,b,c);
}
【题解】SDOI2010所驼门王的宝藏(强连通分量+优化建图)的更多相关文章
- BZOJ1924:[SDOI2010]所驼门王的宝藏(强连通分量,拓扑排序)
Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...
- BZOJ 1924 所驼门王的宝藏(强连通分量缩点+DAG最长链)
思路不是很难,因为宝藏只会在给出的n个点内有,于是只需要在这n个点里面连边,一个点如果能到达另一个点则连一条有向边, 这样用强连通分量缩点后答案就是DAG的最长链. 问题在于暴力建图是O(n^2)的, ...
- 题解 [SDOI2010]所驼门王的宝藏
传送门 保分题再度爆零,自闭ing×2 tarjan没写vis数组,点权算的也有点问题 这题情况3的连边有点麻烦,考场上想了暴力想了二分就是没想到可以直接拿map水过去 不过map果然贼慢,所以这也是 ...
- [BZOJ 1924][Sdoi2010]所驼门王的宝藏
1924: [Sdoi2010]所驼门王的宝藏 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 1285 Solved: 574[Submit][Sta ...
- 洛谷 2403 [SDOI2010] 所驼门王的宝藏
题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为“先知”的Alpaca L. Sotomon是这个家族的领袖,外人也称其为“所驼门王”.所驼门王毕生致力于维护家族的安定与和谐, ...
- [SDOI2010]所驼门王的宝藏
题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为"先知"的Alpaca L. Sotomon是这个家族的领袖,外人也称其为"所驼门王". ...
- [LuoguP2403][SDOI2010]所驼门王的宝藏
题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为"先知"的Alpaca L. Sotomon是这个家族的领袖,外人也称其为"所驼门王". ...
- BZOJ 1924: [Sdoi2010]所驼门王的宝藏 【tarjan】
Description 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为“先 知”的Alpaca L. Sotomon 是这个家族的领袖,外人也称其为“所驼门王”.所 驼门王毕生致力于维 ...
- 洛谷 P2403 [SDOI2010]所驼门王的宝藏 题解
题目描述 分析 先放一张图便于理解 这一道题如果暴力建图会被卡成\(n^{2}\) 实际上,在我们暴力建图的时候,有很多边都是重复的 假如一行当中有许多横天门的话,我们就不必要把这一行当中的所有点和每 ...
随机推荐
- 阿里云亮相2019联通合作伙伴大会,边缘计算等3款云产品助力5G时代产业数字化转型
4月23日,2019中国联通合作伙伴大会在上海正式开幕,本次大会以“合作不设限,共筑新生态”为主题,涉及5G.边缘计算.云计算.物联网.新媒体.人工智能.互联网化等各领域超过600家合作伙伴与3万名各 ...
- 伪静态的实现方法:IIS环境下配置
URL 静态化可以提高搜索引擎抓取,开启本功能需要对 Web 服务器增加相应的 Rewrite 规则,且会轻微增加服务器负担.本教程讲解如何在 IIS 环境下配置各个产品的 Rewrite 规则. 下 ...
- Refs
一.The ref callback attribute ref:reference,父组件引用子组件 组件并不是真实的 DOM节点,而是存在于内存之中的一种数据结构,叫做虚拟DOM.只有当它插入文档 ...
- Win7如何显示文件后缀
有些时候,我们需要修改文件的后缀名,但是Windows7系统默认不显示文件后缀.我们怎样显示和修改文件后缀呢?请接着往下看. 工具/原料 一个win7系统 方法/步骤 1 如图所示,此时是无法 ...
- H3C 专线连接模型
- poj 2451 Uyuw's Concert (半平面交)
2451 -- Uyuw's Concert 继续半平面交,这还是简单的半平面交求面积,不过输入用cin超时了一次. 代码如下: #include <cstdio> #include &l ...
- 【Activiti工作流引擎】官方快速入门demo
Activiti官方快速入门demo 地址: https://www.activiti.org/quick-start 0. 版本 activiti 5.22.0 JDK 1.8 1. 介绍 这个快速 ...
- hdu 1754 I Hate It(线段树区间求最值)
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- 在eclipse中建立lua开发环境
1. 给你的eclipse安装LuaEclipse,新增Eclipse Software Update Site“http://luaeclipse.luaforge.net/update-site ...
- CSS 实现单行及多行文字省略
单行文字省略 很多时候不确定字数限制,但换行可能影响整体设计,这个时候常用就是文字省略加全文字提示了 .dom{ text-overflow: ellipsis; overflow: hidden; ...