「BZOJ 1924」「SDOI 2010」所驼门王的宝藏「Tarjan」
题意
一个\(r\times c\)的棋盘,棋盘上有\(n\)个标记点,每个点有三种类型,类型\(1\)可以传送到本行任意标记点,类型\(2\)可以传送到本列任意标记点,类型\(3\)可以传送到周围八连通任意标记点。求最长路径。
\(r,c\leq 10^6,n\leq 10^5\)
题解
这题做法很多,我就把每一行的所有类型\(1\)门缩到一起(直接找一个代表),列也同理,然后暴力连边,类型\(3\)连边用\(\text{map}\),这样每个点的入边中类型\(1\)或\(2\)最多有\(1\)条,类型\(3\)最多\(8\)条,大概可以说明边数和点数同阶,于是\(\text{Tarjan}\)缩点然后\(dp\)求最长路...
#include <algorithm>
#include <utility>
#include <cstdio>
#include <vector>
#include <stack>
#include <map>
using namespace std;
const int N = 1e5 + 10;
const int dx[] = {1, 0, -1, 0, 1, 1, -1, -1};
const int dy[] = {0, 1, 0, -1, -1, 1, -1, 1};
struct node {
int x, y, z, sz;
} a[N];
int n, r, c, f[N], rt[2][N * 10], dT[N];
vector<int> ob[2][N * 10], G[N], T[N];
map<pair<int, int>, int> ma;
bool isr[N];
int dfn[N], low[N], sz[N], bel[N], scc;
stack<int> st;
bool ins[N];
void tarjan(int u) {
low[u] = dfn[u] = ++ dfn[0];
st.push(u); ins[u] = 1;
for(int i = 0; i < G[u].size(); i ++) {
int v = G[u][i];
if(!dfn[v]) {
tarjan(v);
low[u] = min(low[u], low[v]);
} else if(ins[v]) {
low[u] = min(low[u], dfn[v]);
}
}
if(low[u] == dfn[u]) {
scc ++;
while(1) {
int v = st.top(); st.pop();
ins[v] = 0; bel[v] = scc;
sz[scc] += a[v].sz;
if(u == v) break ;
}
}
}
int pa[N];
int solve(int u) {
if(pa[u]) return pa[u];
for(int i = 0; i < T[u].size(); i ++) {
pa[u] = max(pa[u], solve(T[u][i]));
}
pa[u] += sz[u];
return pa[u];
}
int main() {
scanf("%d%d%d", &n, &r, &c);
for(int i = 1; i <= n; i ++) {
scanf("%d%d%d", &a[i].x, &a[i].y, &a[i].z);
ma[make_pair(a[i].x, a[i].y)] = i;
ob[0][a[i].x].push_back(i);
ob[1][a[i].y].push_back(i);
a[i].sz = 0;
}
for(int i = 1; i <= n; i ++) {
if(a[i].z == 1) {
int &u = rt[0][a[i].x];
if(!u) u = i;
a[u].sz ++; f[i] = u;
}
if(a[i].z == 2) {
int &u = rt[1][a[i].y];
if(!u) u = i;
a[u].sz ++; f[i] = u;
}
if(a[i].z == 3) {
f[i] = i;
a[i].sz ++;
}
isr[f[i]] = 1;
}
for(int i = 1; i <= n; i ++)
if(isr[i]) {
if(a[i].z == 1) {
for(int j = 0; j < ob[0][a[i].x].size(); j ++) {
int v = ob[0][a[i].x][j];
G[i].push_back(f[v]);
}
}
if(a[i].z == 2) {
for(int j = 0; j < ob[1][a[i].y].size(); j ++) {
int v = ob[1][a[i].y][j];
G[i].push_back(f[v]);
}
}
if(a[i].z == 3) {
for(int j = 0; j < 8; j ++) {
int v = ma[make_pair(a[i].x + dx[j], a[i].y + dy[j])];
if(v) {
G[i].push_back(f[v]);
}
}
}
}
for(int i = 1; i <= n; i ++)
if(isr[i] && !dfn[i]) {
tarjan(i);
}
for(int i = 1; i <= n; i ++) {
if(isr[i]) {
for(int j = 0; j < G[i].size(); j ++) {
int v = G[i][j];
if(bel[i] != bel[v]) {
T[bel[i]].push_back(bel[v]);
dT[bel[v]] ++;
}
}
}
}
int ans = 0;
for(int i = 1; i <= scc; i ++)
if(!dT[i]) {
ans = max(ans, solve(i));
}
printf("%d\n", ans);
return 0;
}
「BZOJ 1924」「SDOI 2010」所驼门王的宝藏「Tarjan」的更多相关文章
- BZOJ 1924: [Sdoi2010]所驼门王的宝藏 【tarjan】
Description 在宽广的非洲荒漠中,生活着一群勤劳勇敢的羊驼家族.被族人恭称为“先 知”的Alpaca L. Sotomon 是这个家族的领袖,外人也称其为“所驼门王”.所 驼门王毕生致力于维 ...
- 「BZOJ1924」「SDOI2010」 所驼门王的宝藏 tarjan + dp(DAG 最长路)
「BZOJ1924」[SDOI2010] 所驼门王的宝藏 tarjan + dp(DAG 最长路) -------------------------------------------------- ...
- [BZOJ 1924][Sdoi2010]所驼门王的宝藏
1924: [Sdoi2010]所驼门王的宝藏 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 1285 Solved: 574[Submit][Sta ...
- 「模拟8.18」字符串(卡特兰数)·乌鸦喝水(树状数组,二分)·所驼门王的宝藏(tarjan,拓扑)
最近好颓啊,所以啥都做不出来 简单说一下这次考试,分机房了,还分不同考卷,果然我还是留在二机房的蒟蒻, 大概也只有这样的简单题,才能勉强水个rank 3吧........ 其实不必管在哪个机房,努力便 ...
- bzoj 1924 [Sdoi2010]所驼门王的宝藏(构图,SCC,DP)
Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...
- 【刷题】BZOJ 1924 [Sdoi2010]所驼门王的宝藏
Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...
- BZOJ 1924 所驼门王的宝藏(强连通分量缩点+DAG最长链)
思路不是很难,因为宝藏只会在给出的n个点内有,于是只需要在这n个点里面连边,一个点如果能到达另一个点则连一条有向边, 这样用强连通分量缩点后答案就是DAG的最长链. 问题在于暴力建图是O(n^2)的, ...
- 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 ...
- 所驼门王的宝藏(bzoj 1924)
Description Input 第一行给出三个正整数 N, R, C. 以下 N 行,每行给出一扇传送门的信息,包含三个正整数xi, yi, Ti,表示该传送门设在位于第 xi行第yi列的藏宝宫室 ...
随机推荐
- 线程--demo3
示例1:SwingAndThread package com.etc.jichu; import java.awt.Container; import java.net.URL; import jav ...
- 【转】Context.getExternalFilesDir()和Context.getExternalCacheDir()方法
应用程序在运行的过程中如果需要向手机上保存数据,一般是把数据保存在SDcard中的.大部分应用是直接在SDCard的根目录下创建一个文件夹,然后把数据保存在该文件夹中.这样当该应用被卸载后,这些数据还 ...
- IDA Pro 权威指南学习笔记(四) - IDA 用户界面的基本规则
基本规则: IDA 不提供撤销功能 如果由于不小心按下某个键,导致数据库文件发生意外,这时需要将显示窗口恢复到以前的状态 几乎所有的操作都有其对应的菜单项.热键和工具栏按钮 IDA 的工具栏高度可配置 ...
- NetBeans+Xdebug调试原理
使用Xdebug的远程调试,Xdebug作为一个嵌入到PHP的程序,扮演着客户端的角色,而IDE则作为服务器.下面的动态图展示了连接建立的过程. 服务端的IP为10.0.1.2, 使用HTTP协议,端 ...
- Zookeeper Api(java)入门与应用
如何使用 Zookeeper 作为一个分布式的服务框架,主要用来解决分布式集群中应用系统的一致性问题,它能提供基于类似于文件系统的目录节点树方式的数据存储,但是 Zookeeper 并不是用来专门存储 ...
- Ant之build.xml详解---可用
Ant的概念 :在Eclipse中使用Ant Ant是Java平台下非常棒的批处理命令执行程序,能非常方便地自动完成编译,测试,打包,部署等等一系列任务,大大提高开发效率. Ant和make命令很像. ...
- Win10 Tensorflow 配置Mask_RCNN
1.安装Anaconda3 下载地址 Anaconda 官网下载地址:https://www.continuum.io/downloads 下载以后,点击exe程序,开始安装,详细的安装过程(图片参 ...
- Java Http 请求
package zr.weixin.com.utils; import java.io.BufferedReader; import java.io.IOException; import java. ...
- Hadoop完全分布式环境搭建(二)——基于Ubuntu16.04设置免密登录
在Windows里,使用虚拟机软件Vmware WorkStation搭建三台机器,操作系统Ubuntu16.04,下面是IP和机器名称. [实验目标]:在这三台机器之间实现免密登录 1.从主节点可以 ...
- 数据库 MySQL 之 数据操作
数据库 MySQL 之 数据操作 一.MySQL数据类型介绍 MySQL支持多种类型,大致可以分为四类:数值.字符串类型.日期/时间和其他类型. ①二进制类型 bit[(M)] 二进制位(101001 ...