【洛谷4219】[BJOI2014]大融合(线段树分治)
题目:
分析:
很明显,查询的是删掉某条边后两端点所在连通块大小的乘积。
有加边和删边,想到LCT。但是我不会用LCT查连通块大小啊。果断弃了
有加边和删边,还跟连通性有关,于是开始yy线段树分治做法(不知道线段树分治?推荐一个伪模板题BZOJ4025二分图事实上这个链接是指向我的博客的)。把每次操作(加边或查询)看做一个时刻,一条边存在的区间就是它加入后没有被查询的时间区间的并。于是用可撤销并查集维护一下连通块大小即可。
代码:
#include <cstdio>
#include <cstring>
#include <cctype>
#include <algorithm>
#include <map>
#include <cassert>
#undef i
#undef j
#undef k
#undef min
#undef max
#undef true
#undef false
#undef swap
#undef sort
#undef if
#undef for
#undef while
#undef printf
#undef scanf
#undef putchar
#undef getchar
#define _ 0
using namespace std;
namespace zyt
{
template<typename T>
inline bool read(T &x)
{
char c;
bool f = false;
x = 0;
do
c = getchar();
while (c != EOF && c != '-' && !isdigit(c));
if (c == EOF)
return false;
if (c == '-')
f = true, c = getchar();
do
x = x * 10 + c - '0', c = getchar();
while (isdigit(c));
if (f)
x = -x;
return true;
}
inline bool read(char &c)
{
do
c = getchar();
while (c != EOF && !isgraph(c));
return c != EOF;
}
template<typename T>
inline void write(T x)
{
static char buf[20];
char *pos = buf;
if (x < 0)
putchar('-'), x = -x;
do
*pos++ = x % 10 + '0';
while (x /= 10);
while (pos > buf)
putchar(*--pos);
}
typedef long long ll;
const int N = 1e5 + 10, B = 17, QUERY = 0, ADD = 1;
int n, q;
ll ans[N];
struct node
{
bool type;
int x, y;
}arr[N];
namespace UFS
{
int fa[N], rk[N], size[N], top;
struct node
{
int x, y, fax, rky, sizey;
}stack[N];
void init(const int n)
{
for (int i = 1; i <= n; i++)
fa[i] = i, rk[i] = size[i] = 1;
}
int f(const int u)
{
return fa[u] == u ? u : f(fa[u]);
}
bool merge(const int u, const int v)
{
int x = f(u), y = f(v);
if (x == y)
return false;
if (rk[x] > rk[y])
swap(x, y);
stack[top++] = (node){x, y, fa[x], rk[y], size[y]};
fa[x] = y, size[y] += size[x];
if (rk[x] == rk[y])
++rk[y];
return true;
}
int query(const int u)
{
assert(f(u) < N);
return size[f(u)];
}
void undo(const int bck)
{
while (top > bck)
{
--top;
int x = stack[top].x, y = stack[top].y;
assert(x < N && y < N);
fa[x] = stack[top].fax;
rk[y] = stack[top].rky;
size[y] = stack[top].sizey;
}
}
}
namespace Segment_Tree
{
struct edge
{
int x, y, next;
}e[N * (B + 1)];
int head[1 << (B + 1) | 11], ecnt;
inline void init()
{
memset(head, -1, sizeof(head));
}
inline void add(const int a, const int b, const int c)
{
e[ecnt] = (edge){b, c, head[a]}, head[a] = ecnt++;
}
inline void insert(const int rot, const int lt, const int rt, const int ls, const int rs, const int x, const int y)
{
if (ls <= lt && rt <= rs)
{
add(rot, x, y);
return;
}
int mid = (lt + rt) >> 1;
if (ls <= mid)
insert(rot << 1, lt, mid, ls, rs, x, y);
if (rs > mid)
insert(rot << 1 | 1, mid + 1, rt, ls, rs, x, y);
}
inline void solve(const int rot, const int lt, const int rt)
{
int bck = UFS::top;
for (int i = head[rot]; ~i; i = e[i].next)
UFS::merge(e[i].x, e[i].y);
if (lt == rt)
{
if (arr[lt].type == QUERY)
ans[lt] = (ll)UFS::query(arr[lt].x) * UFS::query(arr[lt].y);
UFS::undo(bck);
return;
}
int mid = (lt + rt) >> 1;
solve(rot << 1, lt, mid);
solve(rot << 1 | 1, mid + 1, rt);
UFS::undo(bck);
}
}
map<pair<int, int>, int> lastins;
int work()
{
read(n), read(q);
UFS::init(n);
Segment_Tree::init();
for (int i = 1; i <= q; i++)
{
char opt;
read(opt), read(arr[i].x), read(arr[i].y);
if (arr[i].x > arr[i].y)
swap(arr[i].x, arr[i].y);
arr[i].type = (opt == 'Q' ? QUERY : ADD);
pair<int, int> p = make_pair(arr[i].x, arr[i].y);
if (arr[i].type == ADD)
lastins[p] = i;
else
{
Segment_Tree::insert(1, 1, q, lastins[p], i - 1, p.first, p.second);
lastins[p] = i + 1;
}
}
for (map<pair<int, int>, int>::iterator it = lastins.begin(); it != lastins.end(); it++)
if (it->second <= q)
Segment_Tree::insert(1, 1, q, it->second, q, it->first.first, it->first.second);
Segment_Tree::solve(1, 1, q);
for (int i = 1; i <= q; i++)
if (arr[i].type == QUERY)
write(ans[i]), putchar('\n');
return (0^_^0);
}
}
int main()
{
return zyt::work();
}
【洛谷4219】[BJOI2014]大融合(线段树分治)的更多相关文章
- 洛谷4219 BJOI2014大融合(LCT维护子树信息)
QWQ 这个题目是LCT维护子树信息的经典应用 根据题目信息来看,对于一个这条边的两个端点各自的\(size\)乘起来,不过这个应该算呢? 我们可以考虑在LCT上多维护一个\(xv[i]\)表示\(i ...
- 洛谷 P4219 [BJOI2014]大融合 解题报告
P4219 [BJOI2014]大融合 题目描述 小强要在\(N\)个孤立的星球上建立起一套通信系统.这套通信系统就是连接\(N\)个点的一个树. 这个树的边是一条一条添加上去的.在某个时刻,一条边的 ...
- 洛谷P4219 - [BJOI2014]大融合
Portal Description 初始有\(n(n\leq10^5)\)个孤立的点,进行\(Q(Q\leq10^5)\)次操作: 连接边\((u,v)\),保证\(u,v\)不连通. 询问有多少条 ...
- 洛谷 P4219 [BJOI2014]大融合
查询,就相当于先删去这条边,然后查询边的两个端点所在连通块大小,乘起来得到答案,然后再把边加回去 可以用线段树分治做 #pragma GCC optimize("Ofast") # ...
- 洛谷 P2147 [SDOI2008]洞穴勘测 (线段树分治)
题目链接 题解 早就想写线段树分治的题了. 对于每条边,它存在于一段时间 我们按时间来搞 我们可把一条边看做一条线段 我们可以模拟线段树操作,不断分治下去 把覆盖\(l-r\)这段时间的线段筛选出来, ...
- 洛谷P4219 [BJOI2014]大融合(LCT,Splay)
LCT维护子树信息的思路总结与其它问题详见我的LCT总结 思路分析 动态连边,LCT题目跑不了了.然而这题又有点奇特的地方. 我们分析一下,查询操作就是要让我们求出砍断这条边后,x和y各自子树大小的乘 ...
- 洛谷P4219 [BJOI2014]大融合(LCT)
LCT维护子树信息的思路总结与其它问题详见我的LCT总结 思路分析 动态连边,LCT题目跑不了了.然而这题又有点奇特的地方. 我们分析一下,查询操作就是要让我们求出砍断这条边后,x和y各自子树大小的乘 ...
- 洛谷 P3373 【模板】线段树 2
洛谷 P3373 [模板]线段树 2 洛谷传送门 题目描述 如题,已知一个数列,你需要进行下面三种操作: 将某区间每一个数乘上 xx 将某区间每一个数加上 xx 求出某区间每一个数的和 输入格式 第一 ...
- 【BZOJ-4530】大融合 线段树合并
4530: [Bjoi2014]大融合 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 280 Solved: 167[Submit][Status] ...
- 洛谷P4344 脑洞治疗仪 [SHOI2015] 线段树+二分答案/分块
!!!一道巨恶心的数据结构题,做完当场爆炸:) 首先,如果你用位运算的时候不小心<<打成>>了,你就可以像我一样陷入疯狂的死循环改半个小时 然后,如果你改出来之后忘记把陷入死循 ...
随机推荐
- Servlet的说明及使用案例
Servlet的说明及使用案例 制作人:全心全意 Servle的基础介绍 Servlet结构体系 Servlet对象.ServletConfig对象与Serializable对象是接口对象,其中Ser ...
- Python selenium chrome打包exe后禁用控制台输出滚动日志
Python selenium chrome打包exe后,在运行的过程中,如果遇到需要input()输入时,会发现被不断滚动刷新的日志把命令行输入快速顶掉了,通过查阅资料不断实践,发现以下方法有效: ...
- SQL学习笔记:表的约束
目录 NOT NULL约束 INDEX 索引 CHECK 约束 DEFAULT 约束 UNIQUE 约束 PRIMARY KEY 约束 FOREIGN KEY 约束:简单的说,就是创建表的时候,对表或 ...
- 【Codeforces 979B】Treasure Hunt
[链接] 我是链接,点我呀:) [题意] 每次你可以将一个字符变成一个不同于本身的字符. 每个人需要改变n次(且不能不改变) 设每个人的字符串中出现次数最多的字符出现的次数为cnt[0~2] 问你谁的 ...
- VMWare学习总结(1)——Centos7安装完毕后无法联网的解决方法
在VmWare 上安装Centos7时,装好vmware后还是连不上网,通过查找资料原来是因为有线网卡没有激活,默认centos和redhat7都是不启用有线网卡的,要么手动开启,要么安装时直接启用! ...
- 使用Mybatis进行连表查询、left join---https://blog.csdn.net/jinzhencs/article/details/51980518
使用Mybatis进行连表查询.left join https://blog.csdn.net/jinzhencs/article/details/51980518
- sdibt 1251 进化树问题
/* 三个点的话 A--D--B | C dis(AD)=(AB+AC-BC)/2; 拓展到到n个点 每次去叶子节点,先去掉与A相连长度最小的. 将他们的长度加起来. */ #include<s ...
- redis+spring
1. 在配置文件中添加 注解 <cache:annotation-driven cache-manager="cacheManager" key-generator=&quo ...
- ubuntu 16.04网卡找不到eth0
自15版本开始就不叫eth0.可以通过ifconfig进行查看: ifconfig -a 其中enp3s0才是网卡的名称,lo为环路. 参考: http://blog.csdn.net/christn ...
- 《WF in 24 Hours》读书笔记 - Hour 2(2) - Workflow:创建分支和传递参数
1. 修改workflow1.xaml,添加Flowchart,添加Flowdesicion,添加CodeActivity2,并且名字改为下图所示: 2. CodeActivity1和2的代码如下: ...