【题解】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}\) 实际上,在我们暴力建图的时候,有很多边都是重复的 假如一行当中有许多横天门的话,我们就不必要把这一行当中的所有点和每 ...
随机推荐
- Hbase数据模型 时间戳
- C#中的?操作符
一.1个?的用法 1. 表示可空数据类型,如 int? bool? 2. 跟在对象后,如该对象为null,则不会触发空值异常,且整个表达式返回null,如: string kk = "123 ...
- 写给初学前端工程师的一封信 - 转载 至https://www.w3ctech.com/topic/983
以下内容是转载https://www.w3ctech.com/topic/983 大家好: 应波波的邀请写一写我对这个话题的想法.从去年开始不少朋友让我帮忙介绍前端工程师,绝大部分忙都没帮上,原因是真 ...
- hdu 1286 找新朋友 (容斥原理 || 欧拉函数)
Problem - 1286 用容斥原理做的代码: #include <cstdio> #include <iostream> #include <algorithm&g ...
- MVC4 变更模板
模板位置: C:\Program Files (x86)\Microsoft Visual Studio 12.0\Common7\IDE\ItemTemplates\CSharp\Web\MVC 4 ...
- get和post的区别?
GET:一般用于信息获取,使用URL传递参数,对所发送信息的数量也有限制,一般在2000个字符 POST:一般用于 修改服务器上的资源,对所发送的信息没有限制. GET方式需要使用Request.Qu ...
- Windows 10 Shared folder - 5168: Spn check for SMB/SMB2 fails.
在搭建 Win10 Shared Folder 时,运行一段时间后就报 Access denied. 导致 Shared Folder 访问不了. 查了下 Event Viewer -> Win ...
- Codeforces Round #597 (Div. 2) D. Shichikuji and Power Grid 题解 最小生成树
题目链接:https://codeforces.com/contest/1245/problem/D 题目大意: 平面上有n座城市,第i座城市的坐标是 \(x[i], y[i]\) , 你现在要给n城 ...
- Java一行代码可声明多个同类变量
Java支持一句语句声明多个同类变量. Example: String a = "Hello", c = "hello"; int x = 5, y = 5;
- Flex AIR应用换肤功能(Android和IOS)
说明 换肤功能,即将整个应用的皮肤都进行更换,其实质,是动态加载swf文件的过程,而这些swf文件则有css文件编译而来. 关于换肤功能,在android和ios系统的实现方式是不同的.主要原因,是因 ...