1924: [Sdoi2010]所驼门王的宝藏

Time Limit: 5 Sec  Memory Limit: 128 MB
Submit: 1285  Solved: 574
[Submit][Status][Discuss]

Description

Input

第一行给出三个正整数 N, R, C。 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室,类型为 Ti。Ti是一个1~3间的整数, 1表示可以传送到第 xi行任意一列的“横天门”,2表示可以传送到任意一行第 yi列的“纵寰门”,3表示可以传送到周围 8格宫室的“自由<x>门”。 保证 1≤xi≤R,1≤yi≤C,所有的传送门位置互不相同。

Output

只有一个正整数,表示你确定的路线所经过不同藏宝宫室的最大数目。

Sample Input

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

Sample Output

9

HINT

测试点编号 N R C 1 16 20 20 2 300 1,000 1,000 3 500 100,000 100,000 4 2,500 5,000 5,000 5 50,000 5,000 5,000 6 50,000 1,000,000 1,000,000 7 80,000 1,000,000 1,000,000 8 100,000 1,000,000 1,000,000 9 100,000 1,000,000 1,000,000 10 100,000 1,000,000 1,000,000

Source

题解

按照题意建好有向边, 跑一遍 $Tarjan$ 缩成 $DAG$ 之后直接 $O(n)$  $DP$ 一遍求最大值即可.

至于快速建边, 可以用 $std::set$ 维护同一行/同一列的所有节点编号, 二维 $std::map$ 维护各个节点的位置. 不过如果所有点都在同一行的话可能会$GG$

参考代码

GitHub

 #include <set>
#include <map>
#include <stack>
#include <vector>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <iostream>
#include <algorithm> const int MAXV=1e5+;
const int MAXE=5e6; struct Edge{
int from;
int to;
Edge* next;
};
Edge E[MAXE];
Edge* head[MAXV];
Edge* top=E; struct Node{
int x;
int y;
int type;
};
Node N[MAXV]; int v;
int c;
int r;
int ans;
int clk;
int scc;
int dp[MAXV];
int dfn[MAXV];
int low[MAXV];
int size[MAXV];
int belong[MAXV]; bool inStack[MAXV]; std::stack<int> s;
std::map< int, std::set<int> > sx,sy;
std::map< int, std::map<int,int> > m; void Build();
void DFS(int);
void Tarjan(int);
void SweepEdge();
void Initialize();
void Insert(int,int); int main(){
Initialize();
Build();
for(int i=;i<=v;i++){
if(dfn[i]==)
Tarjan(i);
}
SweepEdge();
for(int i=;i<=v;i++){
if(dp[i]==)
DFS(i);
}
printf("%d\n",ans);
return ;
} void Initialize(){
scanf("%d%d%d",&v,&r,&c);
for(int i=;i<=v;i++){
scanf("%d%d%d",&N[i].x,&N[i].y,&N[i].type);
sx[N[i].x].insert(i);
sy[N[i].y].insert(i);
(m[N[i].x])[N[i].y]=i;
}
} void DFS(int root){
if(dp[root]!=)
return;
else{
dp[root]=size[root];
int max=;
for(Edge* i=head[root];i!=NULL;i=i->next){
DFS(i->to);
max=std::max(max,dp[i->to]);
}
dp[root]+=max;
ans=std::max(dp[root],ans);
}
} void SweepEdge(){
// puts("SE");
Edge* ed=top;
top=E;
memset(head,,sizeof(head));
for(Edge* i=E;i!=ed;i++){
if(belong[i->from]!=belong[i->to])
Insert(belong[i->from],belong[i->to]);
}
} void Tarjan(int root){
dfn[root]=low[root]=++clk;
inStack[root]=true;
s.push(root);
for(Edge* i=head[root];i!=NULL;i=i->next){
if(dfn[i->to]==){
Tarjan(i->to);
low[root]=std::min(low[root],low[i->to]);
}
else if(inStack[i->to]){
low[root]=std::min(low[root],dfn[i->to]);
}
}
if(dfn[root]==low[root]){
scc++;
int top;
do{
top=s.top();
inStack[top]=false;
size[scc]++;
belong[top]=scc;
s.pop();
}while(top!=root);
}
} void Build(){
for(int k=;k<=v;k++){
if(N[k].type==){
std::map< int, std::set<int> >::iterator px=sx.find(N[k].x);
for(std::set<int>::iterator i=(*px).second.begin();i!=(*px).second.end();++i){
if(k!=*i)
Insert(k,*i);
}
}
else if(N[k].type==){
std::map< int, std::set<int> >::iterator py=sy.find(N[k].y);
for(std::set<int>::iterator i=(*py).second.begin();i!=(*py).second.end();++i){
if(k!=*i)
Insert(k,*i);
}
}
else if(N[k].type==){
for(int dx=-;dx<=;dx++){
for(int dy=-;dy<=;dy++){
if(dx!=||dy!=){
int tmp=m[N[k].x+dx][N[k].y+dy];
if(tmp!=&&tmp!=k){
Insert(k,tmp);
}
}
}
}
}
}
} inline void Insert(int from,int to){
// printf("Insert %d to %d\n",from,to);
top->from=from;
top->to=to;
top->next=head[from];
head[from]=top++;
}

Backup

[BZOJ 1924][Sdoi2010]所驼门王的宝藏的更多相关文章

  1. BZOJ 1924: [Sdoi2010]所驼门王的宝藏 【tarjan】

    Description 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为“先 知”的Alpaca L. Sotomon 是这个家族的领袖,外人也称其为“所驼门王”.所 驼门王毕生致力于维 ...

  2. bzoj 1924 [Sdoi2010]所驼门王的宝藏(构图,SCC,DP)

    Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...

  3. 【刷题】BZOJ 1924 [Sdoi2010]所驼门王的宝藏

    Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...

  4. 【题解】SDOI2010所驼门王的宝藏(强连通分量+优化建图)

    [题解]SDOI2010所驼门王的宝藏(强连通分量+优化建图) 最开始我想写线段树优化建图的说,数据结构学傻了233 虽然矩阵很大,但是没什么用,真正有用的是那些关键点 考虑关键点的类型: 横走型 竖 ...

  5. [SDOI2010]所驼门王的宝藏

    题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为"先知"的Alpaca L. Sotomon是这个家族的领袖,外人也称其为"所驼门王". ...

  6. [LuoguP2403][SDOI2010]所驼门王的宝藏

    题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为"先知"的Alpaca L. Sotomon是这个家族的领袖,外人也称其为"所驼门王". ...

  7. 洛谷 2403 [SDOI2010] 所驼门王的宝藏

    题目描述 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为“先知”的Alpaca L. Sotomon是这个家族的领袖,外人也称其为“所驼门王”.所驼门王毕生致力于维护家族的安定与和谐, ...

  8. Luogu 2403 [SDOI2010]所驼门王的宝藏

    BZOJ 1924 内存要算准,我MLE了两次. 建立$n + r + c$个点,对于一个点$i$的坐标为$(x, y)$,连边$(n + x, i)$和$(n + r + y, i)$,代表这一列和 ...

  9. BZOJ 1924 && Luogu P2403 [SDOI2010]所驼门王的宝藏 恶心建图+缩点DP

    记住:map一定要这么用: if(mp[x[i]+dx[j]].find(y[i]+dy[j])!=mp[x[i]+dx[j]].end()) add(i,mp[x[i]+dx[j]][y[i]+dy ...

随机推荐

  1. poj 1222EXTENDED LIGHTS OUT

    高斯消元的题本质思想一样. 学习网址:http://www.cnblogs.com/rainydays/archive/2011/08/31/2160748.html #include <ios ...

  2. System.Web.Caching.Cache类 缓存 各种缓存依赖(转)

    转自:http://www.cnblogs.com/kissdodog/archive/2013/05/07/3064895.html Cache类,是一个用于缓存常用信息的类.HttpRuntime ...

  3. rails 中 create, new, build, save 的用法以及误区汇总 (转)

    自己很初级,初级的不能再初级,所以初次接触rails的时候,对于里面的create,new,build等方法不是很了解,用的很混乱,导致经常出现不必要的bug,很苦恼,决定,总结一下,结合网上已有资源 ...

  4. CC2530zigbee技术-简介协议栈

    前言 说实话,我喜欢自己的原创,虽然我写得可能简单了,但我觉得自己在写博客的路途上,一点一点地积累知识,我也借鉴别人的东西,特别是在写这篇文章时所使用的是markdownpad2写的,原来我根本就不知 ...

  5. ajax从零基础到实战

    一. 什么是AJAX? ajax是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术. 二. 在项目中怎么运用AJAX? 项目主要文件夹目录有img文件夹,css文件夹,js文件夹,如果你要运 ...

  6. JAVA-1NIO概述

    注意: 转载自并发编程网 – ifeve.com本文链接地址: Java NIO系列教程(一) Java NIO 概述 JAVA-1NIO概述 Java NIO 由以下几个核心部分组成: Channe ...

  7. 如何在service实现弹出对话框

           因为一些需求,我想在service处理后台运行时候,会弹出对话框,但是对话框的建立需要传入Context的值,我试过传入this,也就是service自己的context,还有 传入ge ...

  8. tomcat的8088端口被占用

    打开Dos:windows->输入cmd(想必这个都会) 在黑窗口中输入指令:netstat -ano | findstr 8080       指令的意思是找出占用8080端口的进程pid 再 ...

  9. 以太坊-windows-私有链-搭建(非源码)

    初 接触以太坊,只安装,不讲道理: 1.win10系统,64位 2.以太坊钱包 3.以太坊geth客户端 geth 和 钱包可以到ethfans.org的资料库里下载,那里提供国内镜像和官网地址. 钱 ...

  10. 961 -尺寸2N阵列中的N重复元素

    在一个A大小的数组中2N,有N+1独特的元素,这些元素中的一个重复N次. 返回重复N次的元素. 例1: 输入:[1,2,3,3] 输出:3 例2: 输入:[2,1,2,5,3,2] 输出:2 例3: ...