洛谷P4606 [SDOI2018]战略游戏 【圆方树 + 虚树】
题目链接
题解
两点之间必经的点就是圆方树上两点之间的圆点
所以只需建出圆方树
每次询问建出虚树,统计一下虚树边上有多少圆点即可
还要讨论一下经不经过根\(1\)的情况
P4606
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#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 mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 200005,maxm = 100005,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 h[maxn],ne = 1,hh[maxn],Ne = 1;
struct EDGE{int to,nxt;}ed[maxn << 2],e[maxn << 2];
inline void build(int u,int v){
e[++Ne] = (EDGE){v,hh[u]}; hh[u] = Ne;
e[++Ne] = (EDGE){u,hh[v]}; hh[v] = Ne;
}
inline void add(int u,int v){
ed[++ne] = (EDGE){v,h[u]}; h[u] = ne;
ed[++ne] = (EDGE){u,h[v]}; h[v] = ne;
}
int n,m,N,dfn[maxn],low[maxn],st[maxn],top,cnt;
void dfs(int u,int last){
dfn[u] = low[u] = ++cnt;
st[++top] = u;
for (int k = hh[u],to; k; k = e[k].nxt){
if (k == last) continue;
if (!dfn[to = e[k].to]){
dfs(to,k ^ 1);
low[u] = min(low[u],low[to]);
if (low[to] >= dfn[u]){
++N;
do{add(st[top],N);}while (st[top--] != to);
add(N,u);
}
}
else low[u] = min(low[u],dfn[to]);
}
}
int K,a[maxn],fa[maxn][20],d[maxn][20],Dfn[maxn],dep[maxn],dfc;
void DFS(int u){
Dfn[u] = ++dfc; d[u][0] = (u <= n);
REP(i,19){
fa[u][i] = fa[fa[u][i - 1]][i - 1];
d[u][i] = d[u][i - 1] + d[fa[u][i - 1]][i - 1];
}
Redge(u) if ((to = ed[k].to) != fa[u][0]){
fa[to][0] = u; dep[to] = dep[u] + 1; DFS(to);
}
}
inline int lca(int u,int v){
if (dep[u] < dep[v]) swap(u,v);
for (int i = 0,D = dep[u] - dep[v]; (1 << i) <= D; i++)
if (D & (1 << i)) u = fa[u][i];
if (u == v) return u;
for (int i = 19; ~i; i--)
if (fa[u][i] != fa[v][i]){
u = fa[u][i];
v = fa[v][i];
}
return fa[u][0];
}
inline int getup(int u,int v){
if (u == v) return 0;
int re = -(u <= n);
for (int i = 19; ~i; i--)
if (fa[u][i] && dep[fa[u][i]] >= dep[v]){
re += d[u][i];
u = fa[u][i];
}
return re;
}
inline bool cmp(const int& a,const int& b){
return Dfn[a] < Dfn[b];
}
int pre[maxn];
void work(){
st[top = 1] = 1; int tot = 0,M = K;
sort(a + 1,a + 1 + K,cmp);
for (int i = 1; i <= K; i++){
int u = a[i],p = lca(u,st[top]);
if (u == 1) tot = INF;
if (p == st[top]) {if (u != st[top]) st[++top] = u;}
else {
while (true){
if (dep[p] >= dep[st[top - 1]]){
pre[st[top--]] = p; if (p == 1) tot++;
if (st[top] != p) st[++top] = p,a[++M] = p;
break;
}
pre[st[top]] = st[top - 1];
if (st[top - 1] == 1) tot++;
top--;
}
if (u != st[top]) st[++top] = u;
}
}
while (top > 1){
pre[st[top]] = st[top - 1];
if (st[top - 1] == 1) tot++;
top--;
}
if (tot < INF) a[++M] = 1;
int ans = 0;
for (int i = 1; i <= M; i++){
int u = a[i];
if (u == 1){
if (i > K && tot > 1) ans++;
continue;
}
if (pre[u] != 1 || (pre[u] == 1 && tot > 1)) ans += getup(u,pre[u]);
if (i > K) ans += (u <= n);
pre[u] = 0;
}
printf("%d\n",ans);
}
int main(){
int T = read();
while (T--){
cls(dfn); cls(h); cls(hh);
ne = Ne = 1; cnt = dfc = top = 0;
N = n = read(); m = read();
REP(i,m) build(read(),read());
dfs(1,0); DFS(1);
int q = read();
while (q--){
K = read();
REP(i,K) a[i] = read();
work();
}
}
return 0;
}
P4320
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#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 mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 1000005,maxm = 100005,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 h[maxn],ne = 1,hh[maxn],Ne = 1;
struct EDGE{int to,nxt;}ed[maxn << 2],e[maxn << 2];
inline void build(int u,int v){
e[++Ne] = (EDGE){v,hh[u]}; hh[u] = Ne;
e[++Ne] = (EDGE){u,hh[v]}; hh[v] = Ne;
}
inline void add(int u,int v){
ed[++ne] = (EDGE){v,h[u]}; h[u] = ne;
ed[++ne] = (EDGE){u,h[v]}; h[v] = ne;
}
int n,m,N,val[maxn];
int dfn[maxn],low[maxn],st[maxn],top,cnt;
void dfs(int u,int last){
dfn[u] = low[u] = ++cnt;
st[++top] = u;
for (int k = hh[u],to; k; k = e[k].nxt){
if (k == last) continue;
if (!dfn[to = e[k].to]){
dfs(to,k ^ 1);
low[u] = min(low[u],low[to]);
if (low[to] >= dfn[u]){
++N;
do{add(st[top],N);}while (st[top--] != to);
add(N,u);
}
}
else low[u] = min(low[u],dfn[to]);
}
}
int fa[maxn][20],d[maxn][20],Dfn[maxn],dep[maxn],dfc;
void DFS(int u){
Dfn[u] = ++dfc; d[u][0] = (u <= n);
REP(i,19){
fa[u][i] = fa[fa[u][i - 1]][i - 1];
d[u][i] = d[u][i - 1] + d[fa[u][i - 1]][i - 1];
}
Redge(u) if ((to = ed[k].to) != fa[u][0]){
fa[to][0] = u; dep[to] = dep[u] + 1; DFS(to);
}
}
int dis(int u,int v){
int re = 0;
if (dep[u] < dep[v]) swap(u,v);
for (int i = 0,D = dep[u] - dep[v]; (1 << i) <= D; i++)
if (D & (1 << i)){
re += d[u][i];
u = fa[u][i];
}
if (u == v) return re + 1;
for (int i = 19; ~i; i--)
if (fa[u][i] != fa[v][i]){
re += d[u][i] + d[v][i];
u = fa[u][i];
v = fa[v][i];
}
re += d[u][0] + d[v][0];
if (fa[u][0] <= n) re++;
return re;
}
int main(){
N = n = read(); m = read();
REP(i,m) build(read(),read());
dfs(1,0);
DFS(1);
int q = read(),u,v;
while (q--){
u = read(); v = read();
printf("%d\n",dis(u,v));
}
return 0;
}
洛谷P4606 [SDOI2018]战略游戏 【圆方树 + 虚树】的更多相关文章
- 洛谷P4606 [SDOI2018]战略游戏 [广义圆方树]
传送门 思路 先考虑两点如何使他们不连通. 显然路径上所有的割点都满足条件. 多个点呢?也是这样的. 于是可以想到圆方树.一个点集的答案就是它的虚树里圆点个数减去点集大小. 可以把点按dfs序排序,然 ...
- Luogu P4606 [SDOI2018] 战略游戏 圆方树 虚树
https://www.luogu.org/problemnew/show/P4606 把原来的图的点双联通分量缩点(每个双联通分量建一个点,每个割点再建一个点)(用符合逻辑的方式)建一棵树(我最开始 ...
- bzoj5315/luoguP4517 [SDOI2018]战略游戏(圆方树,虚树)
bzoj5315/luoguP4517 [SDOI2018]战略游戏(圆方树,虚树) bzoj Luogu 题目描述略(太长了) 题解时间 切掉一个点,连通性变化. 上圆方树. $ \sum |S| ...
- [SDOI2018]战略游戏 圆方树,树链剖分
[SDOI2018]战略游戏 这题是道路相遇(题解)的升级版,询问的两个点变成了\(S\)个点. LG传送门 还是先建出圆方树,考虑对于询问的\(S\)个点,答案就是圆方树上能包含这些点的最小连通块中 ...
- 洛谷4606 SDOI2018战略游戏(圆方树+虚树)
QWQ深受其害 当时在现场是真的绝望...... 现在再重新来看这个题 QWQ 根据题目所说,我们可以发现,对于每一个集合中的节点,我们实际上就是要求两两路径上的割点的数目 考虑到又是关于点双的题目, ...
- BZOJ5329:[SDOI2018]战略游戏(圆方树,虚树)
Description 省选临近,放飞自我的小Q无心刷题,于是怂恿小C和他一起颓废,玩起了一款战略游戏. 这款战略游戏的地图由n个城市以及m条连接这些城市的双向道路构成,并且从任意一个城市出发总能沿着 ...
- Luogu4606 SDOI2018 战略游戏 圆方树、虚树、链并
传送门 弱化版 考虑到去掉一个点使得存在两个点不连通的形式类似割点,不难想到建立圆方树.那么在圆方树上对于给出的关键点建立虚树之后,我们需要求的就是虚树路径上所有圆点的数量减去关键点的数量. 因为没有 ...
- BZOJ.5329.[SDOI2018]战略游戏(圆方树 虚树)
题目链接 显然先建圆方树,方点权值为0圆点权值为1,两点间的答案就是路径权值和减去起点终点. 对于询问,显然可以建虚树.但是只需要计算两关键点间路径权值,所以不需要建出虚树.统计DFS序相邻的两关键点 ...
- Solution -「SDOI 2018」「洛谷 P4606」战略游戏
\(\mathcal{Description}\) Link. 给定一个 \(n\) 个点 \(m\) 条边的无向连通图,\(q\) 次询问,每次给出一个点集 \(s\),求至少在原图中删去多 ...
随机推荐
- C# 合并多个结构相同的DataTable
public DataTable GetAllDataTable(DataSet ds) { DataTable newDataTable = ds.Tables[0].Clone(); //创建新表 ...
- python全栈开发-面向对象-初识
python_16_day 函数总结: https://www.processon.com/view/link/5b718274e4b0555b39e1055f 面向过程的程序设计的核心是过程(流水线 ...
- SQL数据类型(SQL Server六个类型使用)
SQL数据类型是一个属性,它指定任何对象的数据的类型.在SQL中每一列,变量和表达有相关数据类型. 当创建表时,需要使用这些数据类型. 会选择根据表列要求选择一个特定的数据类型. SQL Server ...
- zepto 添加 animate组件
今天发现JQuery可以用 animate方法回到顶部,心想着zepto应该也可以 $('html,body').animate({ scrollTop: 0 }, 1000); 于是便用了一下,发现 ...
- 【springmvc+mybatis项目实战】杰信商贸-6.重点知识回顾
1.重点知识回顾 Maven1)覆盖仓库文件,实际企业开发,公司会架一个测试服务器,在测试服务器中架私服.我们开发人员的程序,都连接私服.当本地没有项目中要使用的jar,Myeclipse maven ...
- selenium 图片上传方法。
找到图片上传的input标签: 直接使用send_keys()传值 browser.find_element_by_class_name("upload-pic").send_ke ...
- .net core 2.1.3可能引发Could not load file or assembly XXXXX的错误
参考文档: https://github.com/aspnet/Home/issues/3503 写在前面 感觉自己现在干的活离开发越来越远了啊,不过也很好,每天能学到不少东西,中文的,英文的,永远也 ...
- Paper Reading - Mind’s Eye: A Recurrent Visual Representation for Image Caption Generation ( CVPR 2015 )
Link of the Paper: https://ieeexplore.ieee.org/document/7298856/ A Correlative Paper: Learning a Rec ...
- hbase Problem binding to node1/192.168.1.13:16020 : 地址已在使用
这是hbase 从0.9.x升级到1.x后HMaster与HRegionServer端口冲突问题 在hbase升级到1.0.0版本后,默认端口做了改动.其中16020端口是hmaster服务和hreg ...
- Beta阶段第一次网络会议
Beta阶段第一次网络会议 游戏问题 游戏细节特征不够明显,大小虽然随着电脑分辨率的不同变化着,但是存在清楚的问题 游戏中的提示信息不够,玩家无法快速了解游戏 游戏中背景声音过于单一 游戏AI太简单 ...